Compare commits
20 Commits
v0.108.0-b
...
AGDNS-2743
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6109e3575f | ||
|
|
88706e9cf2 | ||
|
|
b5c47054ab | ||
|
|
4776255604 | ||
|
|
e5d0f0b119 | ||
|
|
af7c2e3a9d | ||
|
|
2c46bc92fe | ||
|
|
61a1403e4e | ||
|
|
4ccc2a2138 | ||
|
|
72425b80a3 | ||
|
|
c7c62ad3b6 | ||
|
|
003e7ce0d5 | ||
|
|
a8fdf1c553 | ||
|
|
7d479baba6 | ||
|
|
feb9c886d8 | ||
|
|
3521e8ed9f | ||
|
|
4d258972d1 | ||
|
|
9726171f0f | ||
|
|
6d282ae716 | ||
|
|
6a99c39d11 |
62
CHANGELOG.md
62
CHANGELOG.md
@@ -9,19 +9,54 @@ The format is based on [*Keep a Changelog*](https://keepachangelog.com/en/1.0.0/
|
|||||||
<!--
|
<!--
|
||||||
## [v0.108.0] – TBA
|
## [v0.108.0] – TBA
|
||||||
|
|
||||||
## [v0.107.60] - 2025-04-01 (APPROX.)
|
## [v0.107.62] - 2025-04-30 (APPROX.)
|
||||||
|
|
||||||
See also the [v0.107.60 GitHub milestone][ms-v0.107.60].
|
See also the [v0.107.62 GitHub milestone][ms-v0.107.62].
|
||||||
|
|
||||||
[ms-v0.107.60]: https://github.com/AdguardTeam/AdGuardHome/milestone/95?closed=1
|
[ms-v0.107.62]: https://github.com/AdguardTeam/AdGuardHome/milestone/97?closed=1
|
||||||
|
|
||||||
NOTE: Add new changes BELOW THIS COMMENT.
|
NOTE: Add new changes BELOW THIS COMMENT.
|
||||||
-->
|
-->
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- DNS cache not working for custom upstream configurations.
|
||||||
|
|
||||||
|
- Validation process for the DNS-over-TLS, DNS-over-QUIC, and HTTPS ports on the *Encryption Settings* page.
|
||||||
|
|
||||||
|
<!--
|
||||||
|
NOTE: Add new changes ABOVE THIS COMMENT.
|
||||||
|
-->
|
||||||
|
|
||||||
|
## [v0.107.61] - 2025-04-22
|
||||||
|
|
||||||
|
See also the [v0.107.61 GitHub milestone][ms-v0.107.61].
|
||||||
|
|
||||||
|
### Security
|
||||||
|
|
||||||
|
- Any simultaneous requests that are considered duplicates will now only result in a single request to upstreams, reducing the chance of a cache poisoning attack succeeding. This is controlled by the new configuration object `pending_requests`, which has a single `enabled` property, set to `true` by default.
|
||||||
|
|
||||||
|
**NOTE:** We thank [Xiang Li][mr-xiang-li] for reporting this security issue. It's strongly recommended to leave it enabled, otherwise AdGuard Home will be vulnerable to untrusted clients.
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Searching for persistent clients using an exact match for CIDR in the `POST /clients/search HTTP API`.
|
||||||
|
|
||||||
|
[mr-xiang-li]: https://lixiang521.com/
|
||||||
|
[ms-v0.107.61]: https://github.com/AdguardTeam/AdGuardHome/milestone/96?closed=1
|
||||||
|
|
||||||
|
## [v0.107.60] - 2025-04-14
|
||||||
|
|
||||||
|
See also the [v0.107.60 GitHub milestone][ms-v0.107.60].
|
||||||
|
|
||||||
### Security
|
### Security
|
||||||
|
|
||||||
- Go version has been updated to prevent the possibility of exploiting the Go vulnerabilities fixed in [1.24.2][go-1.24.2].
|
- Go version has been updated to prevent the possibility of exploiting the Go vulnerabilities fixed in [1.24.2][go-1.24.2].
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Alpine Linux version in `Dockerfile` has been updated to 3.21 ([#7588]).
|
||||||
|
|
||||||
### Deprecated
|
### Deprecated
|
||||||
|
|
||||||
- Node 20 support, Node 22 will be required in future releases.
|
- Node 20 support, Node 22 will be required in future releases.
|
||||||
@@ -32,26 +67,25 @@ NOTE: Add new changes BELOW THIS COMMENT.
|
|||||||
|
|
||||||
- Filtering for DHCP clients ([#7734]).
|
- Filtering for DHCP clients ([#7734]).
|
||||||
|
|
||||||
|
- Incorrect label on login page ([#7729]).
|
||||||
|
|
||||||
- Validation process for the HTTPS port on the *Encryption Settings* page.
|
- Validation process for the HTTPS port on the *Encryption Settings* page.
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
- Node 18 support.
|
- Node 18 support.
|
||||||
|
|
||||||
|
[#7588]: https://github.com/AdguardTeam/AdGuardHome/issues/7588
|
||||||
|
[#7729]: https://github.com/AdguardTeam/AdGuardHome/issues/7729
|
||||||
[#7734]: https://github.com/AdguardTeam/AdGuardHome/issues/7734
|
[#7734]: https://github.com/AdguardTeam/AdGuardHome/issues/7734
|
||||||
|
|
||||||
[go-1.24.2]: https://groups.google.com/g/golang-announce/c/Y2uBTVKjBQk
|
[go-1.24.2]: https://groups.google.com/g/golang-announce/c/Y2uBTVKjBQk
|
||||||
|
[ms-v0.107.60]: https://github.com/AdguardTeam/AdGuardHome/milestone/95?closed=1
|
||||||
<!--
|
|
||||||
NOTE: Add new changes ABOVE THIS COMMENT.
|
|
||||||
-->
|
|
||||||
|
|
||||||
## [v0.107.59] - 2025-03-21
|
## [v0.107.59] - 2025-03-21
|
||||||
|
|
||||||
See also the [v0.107.59 GitHub milestone][ms-v0.107.59].
|
See also the [v0.107.59 GitHub milestone][ms-v0.107.59].
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- Rules with the `client` modifier not working ([#7708]).
|
- Rules with the `client` modifier not working ([#7708]).
|
||||||
|
|
||||||
- The search form not working in the query log ([#7704]).
|
- The search form not working in the query log ([#7704]).
|
||||||
@@ -3092,11 +3126,13 @@ See also the [v0.104.2 GitHub milestone][ms-v0.104.2].
|
|||||||
[ms-v0.104.2]: https://github.com/AdguardTeam/AdGuardHome/milestone/28?closed=1
|
[ms-v0.104.2]: https://github.com/AdguardTeam/AdGuardHome/milestone/28?closed=1
|
||||||
|
|
||||||
<!--
|
<!--
|
||||||
[Unreleased]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.60...HEAD
|
[Unreleased]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.62...HEAD
|
||||||
[v0.107.60]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.59...v0.107.60
|
[v0.107.62]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.61...v0.107.62
|
||||||
-->
|
-->
|
||||||
|
|
||||||
[Unreleased]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.59...HEAD
|
[Unreleased]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.61...HEAD
|
||||||
|
[v0.107.61]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.60...v0.107.61
|
||||||
|
[v0.107.60]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.59...v0.107.60
|
||||||
[v0.107.59]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.58...v0.107.59
|
[v0.107.59]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.58...v0.107.59
|
||||||
[v0.107.58]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.57...v0.107.58
|
[v0.107.58]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.57...v0.107.58
|
||||||
[v0.107.57]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.56...v0.107.57
|
[v0.107.57]: https://github.com/AdguardTeam/AdGuardHome/compare/v0.107.56...v0.107.57
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
# Make sure to sync any changes with the branch overrides below.
|
# Make sure to sync any changes with the branch overrides below.
|
||||||
'variables':
|
'variables':
|
||||||
'channel': 'edge'
|
'channel': 'edge'
|
||||||
'dockerFrontend': 'adguard/home-js-builder:3.0'
|
'dockerFrontend': 'adguard/home-js-builder:3.1'
|
||||||
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
||||||
|
|
||||||
'stages':
|
'stages':
|
||||||
@@ -157,6 +157,7 @@
|
|||||||
|
|
||||||
# Print Docker info.
|
# Print Docker info.
|
||||||
docker info
|
docker info
|
||||||
|
docker buildx version
|
||||||
|
|
||||||
# Prepare and push the build.
|
# Prepare and push the build.
|
||||||
env \
|
env \
|
||||||
@@ -277,7 +278,7 @@
|
|||||||
# need to build a few of these.
|
# need to build a few of these.
|
||||||
'variables':
|
'variables':
|
||||||
'channel': 'beta'
|
'channel': 'beta'
|
||||||
'dockerFrontend': 'adguard/home-js-builder:3.0'
|
'dockerFrontend': 'adguard/home-js-builder:3.1'
|
||||||
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
||||||
# release-vX.Y.Z branches are the branches from which the actual final
|
# release-vX.Y.Z branches are the branches from which the actual final
|
||||||
# release is built.
|
# release is built.
|
||||||
@@ -293,5 +294,5 @@
|
|||||||
# are the ones that actually get released.
|
# are the ones that actually get released.
|
||||||
'variables':
|
'variables':
|
||||||
'channel': 'release'
|
'channel': 'release'
|
||||||
'dockerFrontend': 'adguard/home-js-builder:3.0'
|
'dockerFrontend': 'adguard/home-js-builder:3.1'
|
||||||
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
'key': 'AHBRTSPECS'
|
'key': 'AHBRTSPECS'
|
||||||
'name': 'AdGuard Home - Build and run tests'
|
'name': 'AdGuard Home - Build and run tests'
|
||||||
'variables':
|
'variables':
|
||||||
'dockerFrontend': 'adguard/home-js-builder:3.0'
|
'dockerFrontend': 'adguard/home-js-builder:3.1'
|
||||||
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
||||||
'channel': 'development'
|
'channel': 'development'
|
||||||
|
|
||||||
@@ -233,6 +233,6 @@
|
|||||||
# Set the default release channel on the release branch to beta, as we
|
# Set the default release channel on the release branch to beta, as we
|
||||||
# may need to build a few of these.
|
# may need to build a few of these.
|
||||||
'variables':
|
'variables':
|
||||||
'dockerFrontend': 'adguard/home-js-builder:3.0'
|
'dockerFrontend': 'adguard/home-js-builder:3.1'
|
||||||
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
'dockerGo': 'adguard/go-builder:1.24.2--1'
|
||||||
'channel': 'candidate'
|
'channel': 'candidate'
|
||||||
|
|||||||
1207
client/package-lock.json
generated
vendored
1207
client/package-lock.json
generated
vendored
File diff suppressed because it is too large
Load Diff
4
client/package.json
vendored
4
client/package.json
vendored
@@ -66,7 +66,7 @@
|
|||||||
"@babel/preset-react": "^7.24.1",
|
"@babel/preset-react": "^7.24.1",
|
||||||
"@playwright/test": "1.50.1",
|
"@playwright/test": "1.50.1",
|
||||||
"@types/lodash": "^4.17.4",
|
"@types/lodash": "^4.17.4",
|
||||||
"@types/node": "^22.10.2",
|
"@types/node": "^22.13.10",
|
||||||
"@types/react": "^17.0.80",
|
"@types/react": "^17.0.80",
|
||||||
"@types/react-dom": "^18.3.0",
|
"@types/react-dom": "^18.3.0",
|
||||||
"@types/react-redux": "^7.1.33",
|
"@types/react-redux": "^7.1.33",
|
||||||
@@ -99,7 +99,7 @@
|
|||||||
"stylelint": "^16.5.0",
|
"stylelint": "^16.5.0",
|
||||||
"ts-loader": "^9.5.1",
|
"ts-loader": "^9.5.1",
|
||||||
"url-loader": "^4.1.1",
|
"url-loader": "^4.1.1",
|
||||||
"vitest": "^3.0.4",
|
"vitest": "^3.1.1",
|
||||||
"webpack": "^5.91.0",
|
"webpack": "^5.91.0",
|
||||||
"webpack-cli": "^5.1.4",
|
"webpack-cli": "^5.1.4",
|
||||||
"webpack-dev-server": "^5.0.4",
|
"webpack-dev-server": "^5.0.4",
|
||||||
|
|||||||
@@ -1,24 +1,24 @@
|
|||||||
{
|
{
|
||||||
"client_settings": "Налады кліентаў",
|
"client_settings": "Налады кліентаў",
|
||||||
"example_upstream_reserved": "upstream <0>для канкрэтных даменаў</0>;",
|
"example_upstream_reserved": "upstream <0>для канкрэтных даменаў</0>;",
|
||||||
"example_multiple_upstreams_reserved": "некалькі DNS-сервераў <0>для канкрэтных даменаў</0>;",
|
"example_multiple_upstreams_reserved": "некалькі сервер DNSаў <0>для канкрэтных даменаў</0>;",
|
||||||
"example_upstream_comment": "каментар.",
|
"example_upstream_comment": "каментар.",
|
||||||
"upstream_parallel": "Ужыць адначасныя запыты да ўсіх сервераў для паскарэння апрацоўкі запыту",
|
"upstream_parallel": "Ужыць адначасныя запыты да ўсіх сервераў для паскарэння апрацоўкі запыту",
|
||||||
"parallel_requests": "Паралельныя запыты",
|
"parallel_requests": "Паралельныя запыты",
|
||||||
"load_balancing": "Размеркаванне нагрузкі",
|
"load_balancing": "Размеркаванне нагрузкі",
|
||||||
"load_balancing_desc": "Запытвайце па адным серверы за раз. AdGuard Home будзе выкарыстоўваць выпадковы алгарытм для выбару сервера, так што самы хуткі сервер будзе выкарыстоўвацца часцей.",
|
"load_balancing_desc": "Запытвайце па адным серверы за раз. AdGuard Home будзе выкарыстоўваць выпадковы алгарытм для выбару сервера, так што самы хуткі сервер будзе выкарыстоўвацца часцей.",
|
||||||
"bootstrap_dns": "Bootstrap DNS-серверы",
|
"bootstrap_dns": "Bootstrap сервер DNSы",
|
||||||
"bootstrap_dns_desc": "IP-адрасы DNS-сервераў, якія выкарыстоўваюцца для вырашэння IP-адрасоў распознавальнікаў DoH/DoT, якія вы ўказваеце ў якасці перадачы. Каментары не дапускаюцца.",
|
"bootstrap_dns_desc": "IP-адрасы сервер DNSаў, якія выкарыстоўваюцца для вырашэння IP-адрасоў распознавальнікаў DoH/DoT, якія вы ўказваеце ў якасці перадачы. Каментары не дапускаюцца.",
|
||||||
"fallback_dns_title": "Рэзервовыя DNS-серверы",
|
"fallback_dns_title": "Рэзервовыя сервер DNSы",
|
||||||
"fallback_dns_desc": "Спіс рэзервовых DNS-сервераў, якія выкарыстоўваюцца, калі вышэйшыя DNS-серверы не адказваюць. Сінтаксіс такі ж, як і ў галоўным полі ўверх.",
|
"fallback_dns_desc": "Спіс рэзервовых сервер DNSаў, якія выкарыстоўваюцца, калі вышэйшыя сервер DNSы не адказваюць. Сінтаксіс такі ж, як і ў галоўным полі ўверх.",
|
||||||
"fallback_dns_placeholder": "Увядзіце па адным рэзервовым серверы DNS у радку",
|
"fallback_dns_placeholder": "Увядзіце па адным рэзервовым серверы DNS у радку",
|
||||||
"local_ptr_title": "Прыватныя DNS-серверы",
|
"local_ptr_title": "Прыватныя сервер DNSы",
|
||||||
"local_ptr_desc": "DNS-серверы, якія AdGuard Home выкарыстоўвае для лакальных PTR-запытаў. Гэтыя серверы выкарыстоўваюцца, каб атрымаць даменавыя імёны кліентаў з прыватнымі IP-адрасамі, напрыклад «192.168.12.34», з дапамогай rDNS. Калі спіс пусты, AdGuard Home выкарыстоўвае прадвызначаныя DNS-серверы вашай АС.",
|
"local_ptr_desc": "DNS-серверы, якія AdGuard Home выкарыстоўвае для лакальных PTR-запытаў. Гэтыя серверы выкарыстоўваюцца, каб атрымаць даменавыя імёны кліентаў з прыватнымі IP-адрасамі, напрыклад «192.168.12.34», з дапамогай rDNS. Калі спіс пусты, AdGuard Home выкарыстоўвае прадвызначаныя DNS-серверы вашай АС.",
|
||||||
"local_ptr_default_resolver": "Па змаўчанні AdGuard Home выкарыстоўвае наступныя зваротныя DNS-рэзолверы: {{ip}}.",
|
"local_ptr_default_resolver": "Па змаўчанні AdGuard Home выкарыстоўвае наступныя зваротныя DNS-рэзолверы: {{ip}}.",
|
||||||
"local_ptr_no_default_resolver": "AdGuard Home не змог вызначыць прыдатныя прыватныя адваротныя DNS-рэзолверы для гэтай сістэмы.",
|
"local_ptr_no_default_resolver": "AdGuard Home не змог вызначыць прыдатныя прыватныя адваротныя DNS-рэзолверы для гэтай сістэмы.",
|
||||||
"local_ptr_placeholder": "Увядзіце па адным адрасе на радок",
|
"local_ptr_placeholder": "Увядзіце па адным адрасе на радок",
|
||||||
"resolve_clients_title": "Уключыць запытванне даменавых імёнаў для кліентаў",
|
"resolve_clients_title": "Уключыць запытванне даменавых імёнаў для кліентаў",
|
||||||
"resolve_clients_desc": "AdGuard Home будзе спрабаваць аўтаматычна вызначыць даменавыя імёны кліентаў праз PTR-запыты да адпаведных сервераў (прыватны DNS-сервер для лакальных кліентаў, upstream-серверы для кліентаў з публічным IP-адрасам).",
|
"resolve_clients_desc": "AdGuard Home будзе спрабаваць аўтаматычна вызначыць даменавыя імёны кліентаў праз PTR-запыты да адпаведных сервераў (прыватны сервер DNS для лакальных кліентаў, upstream-серверы для кліентаў з публічным IP-адрасам).",
|
||||||
"use_private_ptr_resolvers_title": "Ужываць прыватныя адваротныя DNS-рэзолверы",
|
"use_private_ptr_resolvers_title": "Ужываць прыватныя адваротныя DNS-рэзолверы",
|
||||||
"use_private_ptr_resolvers_desc": "Пасылаць адваротныя DNS-запыты для лакальна абслугоўных адрасоў на паказаныя серверы. Калі адключана, AdGuard Home будзе адказваць NXDOMAIN на ўсе падобныя PTR-запыты, апроч запытаў пра кліентаў, ужо вядомых па DHCP, /etc/hosts і гэтак далей.",
|
"use_private_ptr_resolvers_desc": "Пасылаць адваротныя DNS-запыты для лакальна абслугоўных адрасоў на паказаныя серверы. Калі адключана, AdGuard Home будзе адказваць NXDOMAIN на ўсе падобныя PTR-запыты, апроч запытаў пра кліентаў, ужо вядомых па DHCP, /etc/hosts і гэтак далей.",
|
||||||
"check_dhcp_servers": "Праверыць DHCP-серверы",
|
"check_dhcp_servers": "Праверыць DHCP-серверы",
|
||||||
@@ -101,13 +101,13 @@
|
|||||||
"compact": "Компактный",
|
"compact": "Компактный",
|
||||||
"nothing_found": "Нічога не знойдзена",
|
"nothing_found": "Нічога не знойдзена",
|
||||||
"faq": "FAQ",
|
"faq": "FAQ",
|
||||||
"version": "версія",
|
"version": "Версія",
|
||||||
"address": "Адрас",
|
"address": "Адрас",
|
||||||
"protocol": "Пратакол",
|
"protocol": "Пратакол",
|
||||||
"on": "УКЛ",
|
"on": "УКЛ",
|
||||||
"off": "Выкл",
|
"off": "Выкл",
|
||||||
"copyright": "Усе правы захаваныя",
|
"copyright": "Усе правы захаваныя",
|
||||||
"homepage": "Галоўная",
|
"homepage": "Хатняя старонка",
|
||||||
"report_an_issue": "Паведаміць пра праблему",
|
"report_an_issue": "Паведаміць пра праблему",
|
||||||
"privacy_policy": "Палітыка прыватнасці",
|
"privacy_policy": "Палітыка прыватнасці",
|
||||||
"enable_protection": "Уключыць абарону",
|
"enable_protection": "Уключыць абарону",
|
||||||
@@ -165,8 +165,8 @@
|
|||||||
"custom_filtering_rules": "Карыстальніцкія правілы фільтрацыі",
|
"custom_filtering_rules": "Карыстальніцкія правілы фільтрацыі",
|
||||||
"encryption_settings": "Налады шыфравання",
|
"encryption_settings": "Налады шыфравання",
|
||||||
"dhcp_settings": "Налады DHCP",
|
"dhcp_settings": "Налады DHCP",
|
||||||
"upstream_dns": "Upstream DNS-серверы",
|
"upstream_dns": "Upstream сервер DNSы",
|
||||||
"upstream_dns_help": "Увядзіце адрасы сервераў па адным у радку. <a>Даведацца больш </a> пра наладжванне DNS-сервераў.",
|
"upstream_dns_help": "Увядзіце адрасы сервераў па адным у радку. <a>Даведацца больш </a> пра наладжванне сервер DNSаў.",
|
||||||
"upstream_dns_configured_in_file": "Наладжаны ў {{path}}",
|
"upstream_dns_configured_in_file": "Наладжаны ў {{path}}",
|
||||||
"test_upstream_btn": "Тэст upstream сервераў",
|
"test_upstream_btn": "Тэст upstream сервераў",
|
||||||
"upstreams": "Upstreams",
|
"upstreams": "Upstreams",
|
||||||
@@ -182,7 +182,7 @@
|
|||||||
"enabled_save_search_toast": "Уключаны бяспечны пошук",
|
"enabled_save_search_toast": "Уключаны бяспечны пошук",
|
||||||
"updated_save_search_toast": "Налады бяспечнага пошуку абноўлены",
|
"updated_save_search_toast": "Налады бяспечнага пошуку абноўлены",
|
||||||
"enabled_table_header": "УКЛ.",
|
"enabled_table_header": "УКЛ.",
|
||||||
"name_table_header": "Імя",
|
"name_table_header": "Назва",
|
||||||
"list_url_table_header": "URL-адрас спіса",
|
"list_url_table_header": "URL-адрас спіса",
|
||||||
"rules_count_table_header": "Колькасць правілаў:",
|
"rules_count_table_header": "Колькасць правілаў:",
|
||||||
"last_time_updated_table_header": "Апошняе абнаўленне",
|
"last_time_updated_table_header": "Апошняе абнаўленне",
|
||||||
@@ -196,7 +196,7 @@
|
|||||||
"no_whitelist_added": "Белыя спісы не дададзены",
|
"no_whitelist_added": "Белыя спісы не дададзены",
|
||||||
"add_blocklist": "Дадаць чорны спіс",
|
"add_blocklist": "Дадаць чорны спіс",
|
||||||
"add_allowlist": "Дадаць белы спіс",
|
"add_allowlist": "Дадаць белы спіс",
|
||||||
"cancel_btn": "Адмена",
|
"cancel_btn": "Скасаваць",
|
||||||
"enter_name_hint": "Увядзіце імя",
|
"enter_name_hint": "Увядзіце імя",
|
||||||
"enter_url_or_path_hint": "Увядзіце URL-адрас ці абсалютны шлях да спіса",
|
"enter_url_or_path_hint": "Увядзіце URL-адрас ці абсалютны шлях да спіса",
|
||||||
"check_updates_btn": "Праверыць абнаўленні",
|
"check_updates_btn": "Праверыць абнаўленні",
|
||||||
@@ -219,7 +219,7 @@
|
|||||||
"example_meaning_host_block": "адказаць 127.0.0.1 для example.org (але не для яго паддаменаў);",
|
"example_meaning_host_block": "адказаць 127.0.0.1 для example.org (але не для яго паддаменаў);",
|
||||||
"example_comment": "! Так можна дадаваць апісанне.",
|
"example_comment": "! Так можна дадаваць апісанне.",
|
||||||
"example_comment_meaning": "каментар;",
|
"example_comment_meaning": "каментар;",
|
||||||
"example_comment_hash": "# І вось так таксама.",
|
"example_comment_hash": "# Таксама каментарый.",
|
||||||
"example_regex_meaning": "блакаваць доступ да даменаў, якія адпавядаюць зададзенаму рэгулярнаму выразу.",
|
"example_regex_meaning": "блакаваць доступ да даменаў, якія адпавядаюць зададзенаму рэгулярнаму выразу.",
|
||||||
"example_upstream_regular": "звычайны DNS (наўзверх UDP);",
|
"example_upstream_regular": "звычайны DNS (наўзверх UDP);",
|
||||||
"example_upstream_regular_port": "звычайны DNS (праз UDP, імя хаста);",
|
"example_upstream_regular_port": "звычайны DNS (праз UDP, імя хаста);",
|
||||||
@@ -233,13 +233,13 @@
|
|||||||
"example_upstream_tcp_port": "звычайны DNS (праз TCP, імя хаста);",
|
"example_upstream_tcp_port": "звычайны DNS (праз TCP, імя хаста);",
|
||||||
"example_upstream_tcp_hostname": "звычайны DNS (праз TCP, імя хаста);",
|
"example_upstream_tcp_hostname": "звычайны DNS (праз TCP, імя хаста);",
|
||||||
"all_lists_up_to_date_toast": "Усе спісы ўжо абноўлены",
|
"all_lists_up_to_date_toast": "Усе спісы ўжо абноўлены",
|
||||||
"updated_upstream_dns_toast": "Upstream DNS-серверы абноўлены",
|
"updated_upstream_dns_toast": "Upstream сервер DNSы абноўлены",
|
||||||
"dns_test_ok_toast": "Паказаныя серверы DNS працуюць карэктна",
|
"dns_test_ok_toast": "Паказаныя серверы DNS працуюць карэктна",
|
||||||
"dns_test_not_ok_toast": "Сервер «{{key}}»: немагчыма выкарыстоўваць, праверце слушнасць напісання",
|
"dns_test_not_ok_toast": "Сервер «{{key}}»: немагчыма выкарыстоўваць, праверце слушнасць напісання",
|
||||||
"dns_test_parsing_error_toast": "Раздзел {{section}}: радок {{line}}: немагчыма выкарыстоўваць, праверце слушнасць напісання",
|
"dns_test_parsing_error_toast": "Раздзел {{section}}: радок {{line}}: немагчыма выкарыстоўваць, праверце слушнасць напісання",
|
||||||
"dns_test_warning_toast": "Upstream «{{key}}» не адказвае на тэставыя запыты і можа не працаваць належным чынам",
|
"dns_test_warning_toast": "Upstream «{{key}}» не адказвае на тэставыя запыты і можа не працаваць належным чынам",
|
||||||
"unblock": "Адблакаваць",
|
"unblock": "Адблакаваць",
|
||||||
"block": "Заблакаваць",
|
"block": "Заблакіраваць",
|
||||||
"disallow_this_client": "Забараніць доступ гэтаму кліенту",
|
"disallow_this_client": "Забараніць доступ гэтаму кліенту",
|
||||||
"allow_this_client": "Дазволіць доступ гэтаму кліенту",
|
"allow_this_client": "Дазволіць доступ гэтаму кліенту",
|
||||||
"block_for_this_client_only": "Заблакаваць толькі для гэтага кліента",
|
"block_for_this_client_only": "Заблакаваць толькі для гэтага кліента",
|
||||||
@@ -259,7 +259,7 @@
|
|||||||
"no_logs_found": "Логі не знойдзены",
|
"no_logs_found": "Логі не знойдзены",
|
||||||
"refresh_btn": "Абнавіць",
|
"refresh_btn": "Абнавіць",
|
||||||
"previous_btn": "Назад",
|
"previous_btn": "Назад",
|
||||||
"next_btn": "Наперад",
|
"next_btn": "Далей",
|
||||||
"loading_table_status": "Загрузка...",
|
"loading_table_status": "Загрузка...",
|
||||||
"page_table_footer_text": "Старонка",
|
"page_table_footer_text": "Старонка",
|
||||||
"rows_table_footer_text": "радкоў",
|
"rows_table_footer_text": "радкоў",
|
||||||
@@ -280,7 +280,7 @@
|
|||||||
"query_log_retention_confirm": "Вы ўпэўнены, што хочаце змяніць тэрмін захоўвання запытаў? Пры памяншэнні інтэрвалу, некаторыя даныя могуць быць страчаны",
|
"query_log_retention_confirm": "Вы ўпэўнены, што хочаце змяніць тэрмін захоўвання запытаў? Пры памяншэнні інтэрвалу, некаторыя даныя могуць быць страчаны",
|
||||||
"anonymize_client_ip": "Ананімізацыя IP-адрасы кліента",
|
"anonymize_client_ip": "Ананімізацыя IP-адрасы кліента",
|
||||||
"anonymize_client_ip_desc": "Не захоўвайце поўныя IP-адрасы гэтых удзельнікаў у часопісах або статыстыцы",
|
"anonymize_client_ip_desc": "Не захоўвайце поўныя IP-адрасы гэтых удзельнікаў у часопісах або статыстыцы",
|
||||||
"dns_config": "Налады DNS-сервера",
|
"dns_config": "Налады сервер DNSа",
|
||||||
"dns_cache_config": "Налада кэша DNS",
|
"dns_cache_config": "Налада кэша DNS",
|
||||||
"dns_cache_config_desc": "Тут можна наладзіць кэш DNS",
|
"dns_cache_config_desc": "Тут можна наладзіць кэш DNS",
|
||||||
"blocking_mode": "Рэжым блакавання",
|
"blocking_mode": "Рэжым блакавання",
|
||||||
@@ -342,14 +342,14 @@
|
|||||||
"unknown_filter": "Невядомы фільтр {{filterId}}",
|
"unknown_filter": "Невядомы фільтр {{filterId}}",
|
||||||
"known_tracker": "Вядомы трэкер",
|
"known_tracker": "Вядомы трэкер",
|
||||||
"install_welcome_title": "Сардэчна запрашаем у AdGuard Home!",
|
"install_welcome_title": "Сардэчна запрашаем у AdGuard Home!",
|
||||||
"install_welcome_desc": "AdGuard Home – гэта DNS-сервер, што блакуе рэкламу і трэкінг. Яго мэта – даць вам магчымасць кантраляваць усю ваша сеціва і ўсе падлучаныя прылады. Ён не патрабуе ўсталёўкі кліенцкіх праграм.",
|
"install_welcome_desc": "AdGuard Home – гэта сервер DNS, што блакуе рэкламу і трэкінг. Яго мэта – даць вам магчымасць кантраляваць усю ваша сеціва і ўсе падлучаныя прылады. Ён не патрабуе ўсталёўкі кліенцкіх праграм.",
|
||||||
"install_settings_title": "Ўэб-інтэрфейс адміністравання",
|
"install_settings_title": "Ўэб-інтэрфейс адміністравання",
|
||||||
"install_settings_listen": "Інтэрфейс сеціва",
|
"install_settings_listen": "Інтэрфейс сеціва",
|
||||||
"install_settings_port": "Порт",
|
"install_settings_port": "Порт",
|
||||||
"install_settings_interface_link": "Ваш ўэб-інтэрфейс адміністравання AdGuard Home будзе даступны па наступных адрасах:",
|
"install_settings_interface_link": "Ваш ўэб-інтэрфейс адміністравання AdGuard Home будзе даступны па наступных адрасах:",
|
||||||
"form_error_port": "Увядзіце карэктны нумар порта",
|
"form_error_port": "Увядзіце карэктны нумар порта",
|
||||||
"install_settings_dns": "DNS-сервер",
|
"install_settings_dns": "DNS-сервер",
|
||||||
"install_settings_dns_desc": "Вам будзе трэба наладзіць свае прылады ці роўтар на выкарыстанне DNS-сервера на адным з наступных адрасоў:",
|
"install_settings_dns_desc": "Вам будзе трэба наладзіць свае прылады ці роўтар на выкарыстанне сервер DNSа на адным з наступных адрасоў:",
|
||||||
"install_settings_all_interfaces": "Усе інтэрфейсы",
|
"install_settings_all_interfaces": "Усе інтэрфейсы",
|
||||||
"install_auth_title": "Аўтарызацыя",
|
"install_auth_title": "Аўтарызацыя",
|
||||||
"install_auth_desc": "Настойліва рэкамендуецца наладзіць аўтэнтыфікацыю паролем для ўэб-інтэрфейсу AdGuard Home. Нават калі ён даступны толькі ў вашай лакальнай сетцы, важна абараніць яго ад неабмежаванага доступу.",
|
"install_auth_desc": "Настойліва рэкамендуецца наладзіць аўтэнтыфікацыю паролем для ўэб-інтэрфейсу AdGuard Home. Нават калі ён даступны толькі ў вашай лакальнай сетцы, важна абараніць яго ад неабмежаванага доступу.",
|
||||||
@@ -365,17 +365,17 @@
|
|||||||
"install_submit_desc": "Працэдура налады завершана і вы гатовы пачаць выкарыстанне AdGuard Home.",
|
"install_submit_desc": "Працэдура налады завершана і вы гатовы пачаць выкарыстанне AdGuard Home.",
|
||||||
"install_devices_router": "Роўтар",
|
"install_devices_router": "Роўтар",
|
||||||
"install_devices_router_desc": "Такая наладка аўтаматычна пакрые ўсе прылады, што выкарыстоўваюць ваш хатні роўтар, і вам не трэба будзе наладжваць кожнае з іх у асобнасці.",
|
"install_devices_router_desc": "Такая наладка аўтаматычна пакрые ўсе прылады, што выкарыстоўваюць ваш хатні роўтар, і вам не трэба будзе наладжваць кожнае з іх у асобнасці.",
|
||||||
"install_devices_address": "DNS-сервер AdGuard Home даступны па наступных адрасах",
|
"install_devices_address": "сервер DNS AdGuard Home даступны па наступных адрасах",
|
||||||
"install_devices_router_list_1": "Адкрыйце налады вашага роўтара. Звычайна вы можаце адкрыць іх у вашым браўзары, напрыклад, http://192.168.0.1/ ці http://192.168.1.1/. Вас могуць папрасіць увесці пароль. Калі вы не помніце яго, пароль часта можна скінуць, націснуўшы на кнопку на самым роўтары. Некаторыя роўтары патрабуюць адмысловага дадатку, які ў гэтым выпадку павінен быць ужо ўсталявана на ваш кампутар ці тэлефон.",
|
"install_devices_router_list_1": "Адкрыйце налады вашага роўтара. Звычайна вы можаце адкрыць іх у вашым браўзары, напрыклад, http://192.168.0.1/ ці http://192.168.1.1/. Вас могуць папрасіць увесці пароль. Калі вы не помніце яго, пароль часта можна скінуць, націснуўшы на кнопку на самым роўтары. Некаторыя роўтары патрабуюць адмысловага дадатку, які ў гэтым выпадку павінен быць ужо ўсталявана на ваш кампутар ці тэлефон.",
|
||||||
"install_devices_router_list_2": "Знайдзіце налады DHCP ці DNS. Знайдзіце літары «DNS» поруч з тэкставым полем, у якое можна ўвесці два ці тры шэрагі лічбаў, падзеленых на 4 групы ад адной до трох лічбаў.",
|
"install_devices_router_list_2": "Знайдзіце налады DHCP ці DNS. Знайдзіце літары «DNS» поруч з тэкставым полем, у якое можна ўвесці два ці тры шэрагі лічбаў, падзеленых на 4 групы ад адной до трох лічбаў.",
|
||||||
"install_devices_router_list_3": "Увядзіце туды адрас вашага AdGuard Home.",
|
"install_devices_router_list_3": "Увядзіце туды адрас вашага AdGuard Home.",
|
||||||
"install_devices_router_list_4": "Вы не можаце ўсталяваць уласны DNS-сервер на некаторых тыпах маршрутызатараў. У гэтым выпадку можа дапамагчы налада AdGuard Home у якасці <a href='#dhcp'>DHCP-сервера</a>. У адваротным выпадку вам трэба звярнуцца да кіраўніцтва па наладзе DNS-сервераў для вашай пэўнай мадэлі маршрутызатара.",
|
"install_devices_router_list_4": "Вы не можаце ўсталяваць уласны сервер DNS на некаторых тыпах маршрутызатараў. У гэтым выпадку можа дапамагчы налада AdGuard Home у якасці <a href='#dhcp'>DHCP-сервера</a>. У адваротным выпадку вам трэба звярнуцца да кіраўніцтва па наладзе сервер DNSаў для вашай пэўнай мадэлі маршрутызатара.",
|
||||||
"install_devices_windows_list_1": "Адкрыйце Панэль кіравання праз меню «Пуск» ці праз пошук Windows.",
|
"install_devices_windows_list_1": "Адкрыйце Панэль кіравання праз меню «Пуск» ці праз пошук Windows.",
|
||||||
"install_devices_windows_list_2": "Перайдзіце ў «Сеціва і інтэрнэт», а потым у «Цэнтр кіравання сеціва і агульным доступам».",
|
"install_devices_windows_list_2": "Перайдзіце ў «Сеціва і інтэрнэт», а потым у «Цэнтр кіравання сеціва і агульным доступам».",
|
||||||
"install_devices_windows_list_3": "У левым боку экрана клікніце «Змена параметраў адаптара».",
|
"install_devices_windows_list_3": "У левым боку экрана клікніце «Змена параметраў адаптара».",
|
||||||
"install_devices_windows_list_4": "Пстрыкніце правай кнопкай мышы ваша актыўнае злучэнне і абярыце Уласцівасці.",
|
"install_devices_windows_list_4": "Пстрыкніце правай кнопкай мышы ваша актыўнае злучэнне і абярыце Уласцівасці.",
|
||||||
"install_devices_windows_list_5": "Знайдзіце ў спісе пункт «IP версіі 4 (TCP/IPv4)», вылучыце яго і потым ізноў націсніце «Уласцівасці».",
|
"install_devices_windows_list_5": "Знайдзіце ў спісе пункт «IP версіі 4 (TCP/IPv4)», вылучыце яго і потым ізноў націсніце «Уласцівасці».",
|
||||||
"install_devices_windows_list_6": "Абярыце «Выкарыстаць наступныя адрасы DNS-сервераў» і ўвядзіце адрас AdGuard Home.",
|
"install_devices_windows_list_6": "Абярыце «Выкарыстаць наступныя адрасы сервер DNSаў» і ўвядзіце адрас AdGuard Home.",
|
||||||
"install_devices_macos_list_1": "Клікніце па абразку Apple і перайдзіце ў Сістэмныя налады.",
|
"install_devices_macos_list_1": "Клікніце па абразку Apple і перайдзіце ў Сістэмныя налады.",
|
||||||
"install_devices_macos_list_2": "Клікніце па іконцы Сеціва.",
|
"install_devices_macos_list_2": "Клікніце па іконцы Сеціва.",
|
||||||
"install_devices_macos_list_3": "Абярыце першае падлучэнне ў спісе і націсніце кнопку «Дадаткова».",
|
"install_devices_macos_list_3": "Абярыце першае падлучэнне ў спісе і націсніце кнопку «Дадаткова».",
|
||||||
@@ -415,7 +415,7 @@
|
|||||||
"encryption_key": "Прыватны ключ",
|
"encryption_key": "Прыватны ключ",
|
||||||
"encryption_key_input": "Скапіюйце сюды прыватны ключ у PEM-кадоўцы.",
|
"encryption_key_input": "Скапіюйце сюды прыватны ключ у PEM-кадоўцы.",
|
||||||
"encryption_enable": "Уключыць шыфраванне (HTTPS, DNS-over-HTTPS і DNS-over-TLS)",
|
"encryption_enable": "Уключыць шыфраванне (HTTPS, DNS-over-HTTPS і DNS-over-TLS)",
|
||||||
"encryption_enable_desc": "Калі шыфраванне ўлучана, ўэб-інтэрфейс AdGuard Home будзе працаваць па HTTPS, а DNS-сервер будзе таксама працаваць па DNS-over-HTTPS і DNS-over-TLS.",
|
"encryption_enable_desc": "Калі шыфраванне ўлучана, ўэб-інтэрфейс AdGuard Home будзе працаваць па HTTPS, а сервер DNS будзе таксама працаваць па DNS-over-HTTPS і DNS-over-TLS.",
|
||||||
"encryption_chain_valid": "Ланцужок сертыфікатаў валідны",
|
"encryption_chain_valid": "Ланцужок сертыфікатаў валідны",
|
||||||
"encryption_chain_invalid": "Ланцужок сертыфікатаў не валідны",
|
"encryption_chain_invalid": "Ланцужок сертыфікатаў не валідны",
|
||||||
"encryption_key_valid": "Валідны {{type}} прыватны ключ",
|
"encryption_key_valid": "Валідны {{type}} прыватны ключ",
|
||||||
@@ -435,8 +435,8 @@
|
|||||||
"update_announcement": "AdGuard Home {{version}} ужо даступная! <0>Націсніце сюды</0>, каб даведацца больш.",
|
"update_announcement": "AdGuard Home {{version}} ужо даступная! <0>Націсніце сюды</0>, каб даведацца больш.",
|
||||||
"setup_guide": "Інструкцыя па наладзе",
|
"setup_guide": "Інструкцыя па наладзе",
|
||||||
"dns_addresses": "Адрасы DNS",
|
"dns_addresses": "Адрасы DNS",
|
||||||
"dns_start": "DNS-сервер запускаецца",
|
"dns_start": "сервер DNS запускаецца",
|
||||||
"dns_status_error": "Памылка праверкі стану DNS-сервера",
|
"dns_status_error": "Памылка праверкі стану сервер DNSа",
|
||||||
"down": "Уніз",
|
"down": "Уніз",
|
||||||
"fix": "Выправіць",
|
"fix": "Выправіць",
|
||||||
"dns_providers": "<0>Спіс вядомых DNS-правайдараў</0> на выбар.",
|
"dns_providers": "<0>Спіс вядомых DNS-правайдараў</0> на выбар.",
|
||||||
@@ -449,7 +449,7 @@
|
|||||||
"settings_global": "Глабальныя",
|
"settings_global": "Глабальныя",
|
||||||
"settings_custom": "Свае",
|
"settings_custom": "Свае",
|
||||||
"table_client": "Кліент",
|
"table_client": "Кліент",
|
||||||
"table_name": "Імя",
|
"table_name": "Назва",
|
||||||
"save_btn": "Захаваць",
|
"save_btn": "Захаваць",
|
||||||
"client_add": "Дадаць кліента",
|
"client_add": "Дадаць кліента",
|
||||||
"client_new": "Новы кліент",
|
"client_new": "Новы кліент",
|
||||||
@@ -475,7 +475,7 @@
|
|||||||
"auto_clients_title": "Кліенты (runtime)",
|
"auto_clients_title": "Кліенты (runtime)",
|
||||||
"auto_clients_desc": "Інфармацыя аб IP-адрасах прылад, якія выкарыстоўваюць або могуць выкарыстоўваць AdGuard Home. Гэтая інфармацыя збіраецца з некалькіх крыніц, уключаючы файлы хостаў, зваротны DNS і г.д.",
|
"auto_clients_desc": "Інфармацыя аб IP-адрасах прылад, якія выкарыстоўваюць або могуць выкарыстоўваць AdGuard Home. Гэтая інфармацыя збіраецца з некалькіх крыніц, уключаючы файлы хостаў, зваротны DNS і г.д.",
|
||||||
"access_title": "Налады доступу",
|
"access_title": "Налады доступу",
|
||||||
"access_desc": "Тут вы можаце наладзіць правілы доступу да DNS-серверу AdGuard Home",
|
"access_desc": "Тут вы можаце наладзіць правілы доступу да сервер DNSу AdGuard Home",
|
||||||
"access_allowed_title": "Дазволеныя кліенты",
|
"access_allowed_title": "Дазволеныя кліенты",
|
||||||
"access_allowed_desc": "Спіс CIDR, IP-адрасоў або <a>ClientID</a>. Калі ў гэтым спісе ёсць запісы, AdGuard Home будзе прымаць запыты толькі ад гэтых кліентаў.",
|
"access_allowed_desc": "Спіс CIDR, IP-адрасоў або <a>ClientID</a>. Калі ў гэтым спісе ёсць запісы, AdGuard Home будзе прымаць запыты толькі ад гэтых кліентаў.",
|
||||||
"access_disallowed_title": "Забароненыя кліенты",
|
"access_disallowed_title": "Забароненыя кліенты",
|
||||||
@@ -596,7 +596,7 @@
|
|||||||
"disable_ipv6_desc": "Ігнараваць усе запыты DNS для адрасоў IPv6 (тып AAAA) і выдаленне дадзеных IPv6 з адказаў тыпу HTTPS.",
|
"disable_ipv6_desc": "Ігнараваць усе запыты DNS для адрасоў IPv6 (тып AAAA) і выдаленне дадзеных IPv6 з адказаў тыпу HTTPS.",
|
||||||
"fastest_addr": "Найхуткі IP-адрас",
|
"fastest_addr": "Найхуткі IP-адрас",
|
||||||
"fastest_addr_desc": "Апытайце ўсе DNS-серверы і вярніце самы хуткі IP-адрас сярод усіх адказаў. Гэта замарудзіць выкананне DNS-запытаў, бо нам давядзецца чакаць адказаў ад усіх DNS-сервераў, але палепшыць агульную ўзаемасувязь.",
|
"fastest_addr_desc": "Апытайце ўсе DNS-серверы і вярніце самы хуткі IP-адрас сярод усіх адказаў. Гэта замарудзіць выкананне DNS-запытаў, бо нам давядзецца чакаць адказаў ад усіх DNS-сервераў, але палепшыць агульную ўзаемасувязь.",
|
||||||
"autofix_warning_text": "Пры націску «Выправіць» AdGuard Home наладзіць вашу сістэму на выкарыстанне DNS-сервера AdGuard Home.",
|
"autofix_warning_text": "Пры націску «Выправіць» AdGuard Home наладзіць вашу сістэму на выкарыстанне сервер DNSа AdGuard Home.",
|
||||||
"autofix_warning_list": "Будуць выконвацца наступныя заданні: <0>Дэактываваць сістэмны DNSStubListener</0> <0>Усталяваць адрас сервера DNS на 127.0.0.1</0> <0>Стварыць сімвалічную спасылку /etc/resolv.conf на /run/systemd/resolve/resolv.conf</0> <0>Спыніць DNSStubListener (перазагрузіць сістэмную службу)</0>.",
|
"autofix_warning_list": "Будуць выконвацца наступныя заданні: <0>Дэактываваць сістэмны DNSStubListener</0> <0>Усталяваць адрас сервера DNS на 127.0.0.1</0> <0>Стварыць сімвалічную спасылку /etc/resolv.conf на /run/systemd/resolve/resolv.conf</0> <0>Спыніць DNSStubListener (перазагрузіць сістэмную службу)</0>.",
|
||||||
"autofix_warning_result": "У выніку ўсе DNS-запыты ад вашай сістэмы будуць па змаўчанні апрацоўвацца AdGuard Home.\n",
|
"autofix_warning_result": "У выніку ўсе DNS-запыты ад вашай сістэмы будуць па змаўчанні апрацоўвацца AdGuard Home.\n",
|
||||||
"tags_title": "Тэгі",
|
"tags_title": "Тэгі",
|
||||||
@@ -634,12 +634,12 @@
|
|||||||
"validated_with_dnssec": "Проверено с помощью DNSSEC",
|
"validated_with_dnssec": "Проверено с помощью DNSSEC",
|
||||||
"all_queries": "Усе запыты",
|
"all_queries": "Усе запыты",
|
||||||
"show_blocked_responses": "Заблакавана",
|
"show_blocked_responses": "Заблакавана",
|
||||||
"show_whitelisted_responses": "Белы спіс",
|
"show_whitelisted_responses": "У белым спісе",
|
||||||
"show_processed_responses": "Апрацавана",
|
"show_processed_responses": "Апрацавана",
|
||||||
"blocked_safebrowsing": "Заблакіравана згодна з базай даных Safe Browsing",
|
"blocked_safebrowsing": "Заблакіравана згодна з базай даных Safe Browsing",
|
||||||
"blocked_adult_websites": "Заблакавана Бацькоўскім кантролем",
|
"blocked_adult_websites": "Заблакавана Бацькоўскім кантролем",
|
||||||
"blocked_threats": "Заблакавана пагроз",
|
"blocked_threats": "Заблакавана пагроз",
|
||||||
"allowed": "Дазволены",
|
"allowed": "У белым спісе",
|
||||||
"filtered": "Адфільтраваныя",
|
"filtered": "Адфільтраваныя",
|
||||||
"rewritten": "Перапісаныя",
|
"rewritten": "Перапісаныя",
|
||||||
"safe_search": "Бяспечны пошук",
|
"safe_search": "Бяспечны пошук",
|
||||||
@@ -738,7 +738,7 @@
|
|||||||
"thursday_short": "Чц.",
|
"thursday_short": "Чц.",
|
||||||
"friday_short": "Пт.",
|
"friday_short": "Пт.",
|
||||||
"saturday_short": "Сб.",
|
"saturday_short": "Сб.",
|
||||||
"upstream_dns_cache_configuration": "Канфігурацыя кэша upstream DNS-сервераў",
|
"upstream_dns_cache_configuration": "Канфігурацыя кэша upstream сервер DNSаў",
|
||||||
"enable_upstream_dns_cache": "Ўключыць кэшаванне для карыстацкай канфігурацыі upstream-сервераў гэтага кліента",
|
"enable_upstream_dns_cache": "Ўключыць кэшаванне для карыстацкай канфігурацыі upstream-сервераў гэтага кліента",
|
||||||
"dns_cache_size": "Памер кэша DNS, у байтах"
|
"dns_cache_size": "Памер кэша DNS, у байтах"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -45,6 +45,7 @@
|
|||||||
"filter": "Филтър",
|
"filter": "Филтър",
|
||||||
"query_log": "История на заявките",
|
"query_log": "История на заявките",
|
||||||
"compact": "Compact",
|
"compact": "Compact",
|
||||||
|
"nothing_found": "Нищо не е намерено",
|
||||||
"faq": "ЧЗВ",
|
"faq": "ЧЗВ",
|
||||||
"version": "версия",
|
"version": "версия",
|
||||||
"address": "Адрес",
|
"address": "Адрес",
|
||||||
@@ -65,14 +66,12 @@
|
|||||||
"stats_malware_phishing": "вируси/атаки",
|
"stats_malware_phishing": "вируси/атаки",
|
||||||
"stats_adult": "сайтове за възрастни",
|
"stats_adult": "сайтове за възрастни",
|
||||||
"stats_query_domain": "Най-отваряни страници",
|
"stats_query_domain": "Най-отваряни страници",
|
||||||
"for_last_24_hours": "за последните 24 часа",
|
|
||||||
"no_domains_found": "Няма намерени резултати",
|
"no_domains_found": "Няма намерени резултати",
|
||||||
"requests_count": "Сума на заявките",
|
"requests_count": "Сума на заявките",
|
||||||
"top_blocked_domains": "Най-блокирани страници",
|
"top_blocked_domains": "Най-блокирани страници",
|
||||||
"top_clients": "Най-активни IP адреси",
|
"top_clients": "Най-активни IP адреси",
|
||||||
"no_clients_found": "Нямa намерени адреси",
|
"no_clients_found": "Нямa намерени адреси",
|
||||||
"general_statistics": "Обща статисика",
|
"general_statistics": "Обща статисика",
|
||||||
"number_of_dns_query_24_hours": "Сума на DNS заявки за последните 24 часа",
|
|
||||||
"number_of_dns_query_blocked_24_hours": "Сума на блокирани DNS заявки от филтрите за реклама и местни",
|
"number_of_dns_query_blocked_24_hours": "Сума на блокирани DNS заявки от филтрите за реклама и местни",
|
||||||
"number_of_dns_query_blocked_24_hours_by_sec": "Сума на блокирани DNS заявки от AdGuard свързани със сигурността",
|
"number_of_dns_query_blocked_24_hours_by_sec": "Сума на блокирани DNS заявки от AdGuard свързани със сигурността",
|
||||||
"number_of_dns_query_blocked_24_hours_adult": "Сума на блокирани сайтове за възрастни",
|
"number_of_dns_query_blocked_24_hours_adult": "Сума на блокирани сайтове за възрастни",
|
||||||
@@ -156,6 +155,7 @@
|
|||||||
"rule_added_to_custom_filtering_toast": "Добавено до местни правила за филтриране: {{rule}}",
|
"rule_added_to_custom_filtering_toast": "Добавено до местни правила за филтриране: {{rule}}",
|
||||||
"default": "По подразбиране",
|
"default": "По подразбиране",
|
||||||
"custom_ip": "Персонализиран IP",
|
"custom_ip": "Персонализиран IP",
|
||||||
|
"dnscrypt": "DNSCrypt",
|
||||||
"dns_over_https": "DNS-пред-HTTPS",
|
"dns_over_https": "DNS-пред-HTTPS",
|
||||||
"dns_over_quic": "DNS-over-QUIC",
|
"dns_over_quic": "DNS-over-QUIC",
|
||||||
"plain_dns": "Обикновен DNS",
|
"plain_dns": "Обикновен DNS",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Zakázaný",
|
"blocklist": "Zakázaný",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Velikost mezipaměti",
|
"cache_size": "Velikost mezipaměti",
|
||||||
"cache_size_desc": "Velikost mezipaměti DNS (v bajtech). Chcete-li ukládání do mezipaměti zakázat, ponechte prázdné.",
|
"cache_size_desc": "Velikost mezipaměti DNS (v bajtech). Chcete-li ukládání do mezipaměti zakázat, nastavte 0.",
|
||||||
"cache_ttl_min_override": "Přepsat minimální hodnotu TTL",
|
"cache_ttl_min_override": "Přepsat minimální hodnotu TTL",
|
||||||
"cache_ttl_max_override": "Přepsat maximální hodnotu TTL",
|
"cache_ttl_max_override": "Přepsat maximální hodnotu TTL",
|
||||||
"enter_cache_size": "Zadejte velikost mezipaměti (v bajtech)",
|
"enter_cache_size": "Zadejte velikost mezipaměti (v bajtech)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Sortliste",
|
"blocklist": "Sortliste",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Cache-størrelse",
|
"cache_size": "Cache-størrelse",
|
||||||
"cache_size_desc": "DNS cache-størrelse (i bytes). Lad stå tomt for at deaktivere cache.",
|
"cache_size_desc": "DNS cache-størrelse (i bytes). Sæt til 0 for at deaktivere cache.",
|
||||||
"cache_ttl_min_override": "Tilsidesæt minimum TTL",
|
"cache_ttl_min_override": "Tilsidesæt minimum TTL",
|
||||||
"cache_ttl_max_override": "Tilsidesæt maksimal TTL",
|
"cache_ttl_max_override": "Tilsidesæt maksimal TTL",
|
||||||
"enter_cache_size": "Angiv cache-størrelse (bytes)",
|
"enter_cache_size": "Angiv cache-størrelse (bytes)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Sperrliste",
|
"blocklist": "Sperrliste",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Größe des Cache",
|
"cache_size": "Größe des Cache",
|
||||||
"cache_size_desc": "Größe des DNS-Zwischenspeichers (in Bytes)",
|
"cache_size_desc": "Größe des DNS-Cache (in Bytes). Um das Caching zu deaktivieren, setzen Sie den Wert auf 0.",
|
||||||
"cache_ttl_min_override": "TTL-Minimalwert überschreiben",
|
"cache_ttl_min_override": "TTL-Minimalwert überschreiben",
|
||||||
"cache_ttl_max_override": "TTL-Höchstwert überschreiben",
|
"cache_ttl_max_override": "TTL-Höchstwert überschreiben",
|
||||||
"enter_cache_size": "Größe des Cache (Bytes) eingeben",
|
"enter_cache_size": "Größe des Cache (Bytes) eingeben",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Blocklist",
|
"blocklist": "Blocklist",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Cache size",
|
"cache_size": "Cache size",
|
||||||
"cache_size_desc": "DNS cache size (in bytes). To disable caching, leave empty.",
|
"cache_size_desc": "DNS cache size (in bytes). To disable caching, set to 0.",
|
||||||
"cache_ttl_min_override": "Override minimum TTL",
|
"cache_ttl_min_override": "Override minimum TTL",
|
||||||
"cache_ttl_max_override": "Override maximum TTL",
|
"cache_ttl_max_override": "Override maximum TTL",
|
||||||
"enter_cache_size": "Enter cache size (bytes)",
|
"enter_cache_size": "Enter cache size (bytes)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Lista de bloqueo",
|
"blocklist": "Lista de bloqueo",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Tamaño de la caché",
|
"cache_size": "Tamaño de la caché",
|
||||||
"cache_size_desc": "Tamaño de la caché DNS (en bytes). Para deshabilitar el almacenamiento en caché, déjalo vacío.",
|
"cache_size_desc": "Tamaño de la caché DNS (en bytes). Para desactivar el almacenamiento en caché, configúralo en 0.",
|
||||||
"cache_ttl_min_override": "Anular TTL mínimo",
|
"cache_ttl_min_override": "Anular TTL mínimo",
|
||||||
"cache_ttl_max_override": "Anular TTL máximo",
|
"cache_ttl_max_override": "Anular TTL máximo",
|
||||||
"enter_cache_size": "Ingresa el tamaño de la caché (bytes)",
|
"enter_cache_size": "Ingresa el tamaño de la caché (bytes)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Liste de blocage",
|
"blocklist": "Liste de blocage",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Taille du cache",
|
"cache_size": "Taille du cache",
|
||||||
"cache_size_desc": "Taille du cache DNS (en octets). Pour désactiver la mise en cache, laissez vide.",
|
"cache_size_desc": "Taille du cache DNS (en octets). Pour désactiver la mise en cache, mettez la valeur sur 0.",
|
||||||
"cache_ttl_min_override": "Remplacer le TTL minimum",
|
"cache_ttl_min_override": "Remplacer le TTL minimum",
|
||||||
"cache_ttl_max_override": "Remplacer le TTL maximum",
|
"cache_ttl_max_override": "Remplacer le TTL maximum",
|
||||||
"enter_cache_size": "Entrer la taille du cache (octets)",
|
"enter_cache_size": "Entrer la taille du cache (octets)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Lista nera",
|
"blocklist": "Lista nera",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Dimensioni cache",
|
"cache_size": "Dimensioni cache",
|
||||||
"cache_size_desc": "Dimensione della cache DNS (in byte). Per disabilitare la memorizzazione nella cache, lascia vuoto.",
|
"cache_size_desc": "Dimensione della cache DNS (in byte). Per disabilitare la cache, impostare su 0.",
|
||||||
"cache_ttl_min_override": "Sovrascrivi TTL minimo",
|
"cache_ttl_min_override": "Sovrascrivi TTL minimo",
|
||||||
"cache_ttl_max_override": "Sovrascrivi TTL massimo",
|
"cache_ttl_max_override": "Sovrascrivi TTL massimo",
|
||||||
"enter_cache_size": "Immetti dimensioni cache (in byte)",
|
"enter_cache_size": "Immetti dimensioni cache (in byte)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "ブロックリスト",
|
"blocklist": "ブロックリスト",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "キャッシュサイズ",
|
"cache_size": "キャッシュサイズ",
|
||||||
"cache_size_desc": "DNSキャッシュサイズ(バイト単位)。※キャッシュを無効化するには、この欄を空してください。",
|
"cache_size_desc": "DNSキャッシュサイズ(バイト単位)※キャッシュを無効化するには、「0」(ゼロ)にしてください。",
|
||||||
"cache_ttl_min_override": "最小TTLの上書き(秒単位)",
|
"cache_ttl_min_override": "最小TTLの上書き(秒単位)",
|
||||||
"cache_ttl_max_override": "最大TTLの上書き(秒単位)",
|
"cache_ttl_max_override": "最大TTLの上書き(秒単位)",
|
||||||
"enter_cache_size": "キャッシュサイズ(バイト単位)を入力してください",
|
"enter_cache_size": "キャッシュサイズ(バイト単位)を入力してください",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "차단 목록",
|
"blocklist": "차단 목록",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "캐시 크기",
|
"cache_size": "캐시 크기",
|
||||||
"cache_size_desc": "DNS 캐시 크기(바이트). 캐싱을 비활성화하려면 비워 둡니다.",
|
"cache_size_desc": "DNS 캐시 크기(바이트). 캐싱을 사용하지 않으려면 0으로 설정합니다.",
|
||||||
"cache_ttl_min_override": "최소 TTL (초) 무시",
|
"cache_ttl_min_override": "최소 TTL (초) 무시",
|
||||||
"cache_ttl_max_override": "최대 TTL (초) 무시",
|
"cache_ttl_max_override": "최대 TTL (초) 무시",
|
||||||
"enter_cache_size": "캐시 크기를 입력하세요",
|
"enter_cache_size": "캐시 크기를 입력하세요",
|
||||||
|
|||||||
@@ -110,9 +110,9 @@
|
|||||||
"homepage": "Startpagina",
|
"homepage": "Startpagina",
|
||||||
"report_an_issue": "Rapporteer een probleem",
|
"report_an_issue": "Rapporteer een probleem",
|
||||||
"privacy_policy": "Privacybeleid",
|
"privacy_policy": "Privacybeleid",
|
||||||
"enable_protection": "Schakel bescherming in",
|
"enable_protection": "Bescherming inschakelen",
|
||||||
"enabled_protection": "Bescherming ingeschakeld",
|
"enabled_protection": "Bescherming ingeschakeld",
|
||||||
"disable_protection": "Schakel bescherming uit",
|
"disable_protection": "Bescherming uitschakelen",
|
||||||
"disabled_protection": "Bescherming uitgeschakeld",
|
"disabled_protection": "Bescherming uitgeschakeld",
|
||||||
"refresh_statics": "Ververs statistieken",
|
"refresh_statics": "Ververs statistieken",
|
||||||
"dns_query": "DNS-queries",
|
"dns_query": "DNS-queries",
|
||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Blokkeerlijst",
|
"blocklist": "Blokkeerlijst",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Cache grootte",
|
"cache_size": "Cache grootte",
|
||||||
"cache_size_desc": "DNS-cachegrootte (in bytes). Leeg laten om caching uit te schakelen.",
|
"cache_size_desc": "DNS-cachegrootte (in bytes). Om caching uit te schakelen, stel deze in op 0.",
|
||||||
"cache_ttl_min_override": "Minimale TTL overschrijven",
|
"cache_ttl_min_override": "Minimale TTL overschrijven",
|
||||||
"cache_ttl_max_override": "Maximale TTL overschrijven",
|
"cache_ttl_max_override": "Maximale TTL overschrijven",
|
||||||
"enter_cache_size": "Cache grootte invoeren (bytes)",
|
"enter_cache_size": "Cache grootte invoeren (bytes)",
|
||||||
@@ -702,13 +702,13 @@
|
|||||||
"disable_for_hours": "Voor {{count}} uur",
|
"disable_for_hours": "Voor {{count}} uur",
|
||||||
"disable_for_hours_plural": "Voor {{count}} uren",
|
"disable_for_hours_plural": "Voor {{count}} uren",
|
||||||
"disable_until_tomorrow": "Tot morgen",
|
"disable_until_tomorrow": "Tot morgen",
|
||||||
"disable_notify_for_seconds": "Beveiliging uitschakelen voor {{count}} seconde",
|
"disable_notify_for_seconds": "Bescherming uitschakelen voor {{count}} seconde",
|
||||||
"disable_notify_for_seconds_plural": "Beveiliging uitschakelen voor {{count}} seconden",
|
"disable_notify_for_seconds_plural": "Bescherming uitschakelen voor {{count}} seconden",
|
||||||
"disable_notify_for_minutes": "Beveiliging uitschakelen voor {{count}} minuut",
|
"disable_notify_for_minutes": "Bescherming uitschakelen voor {{count}} minuut",
|
||||||
"disable_notify_for_minutes_plural": "Beveiliging uitschakelen voor {{count}} minuten",
|
"disable_notify_for_minutes_plural": "Bescherming uitschakelen voor {{count}} minuten",
|
||||||
"disable_notify_for_hours": "Beveiliging uitschakelen voor {{count}} uur",
|
"disable_notify_for_hours": "Bescherming uitschakelen voor {{count}} uur",
|
||||||
"disable_notify_for_hours_plural": "Beveiliging uitschakelen voor {{count}} uren",
|
"disable_notify_for_hours_plural": "Bescherming uitschakelen voor {{count}} uren",
|
||||||
"disable_notify_until_tomorrow": "Beveiliging uitschakelen tot morgen",
|
"disable_notify_until_tomorrow": "Bescherming uitschakelen tot morgen",
|
||||||
"enable_protection_timer": "Bescherming wordt ingeschakeld over {{time}}",
|
"enable_protection_timer": "Bescherming wordt ingeschakeld over {{time}}",
|
||||||
"custom_retention_input": "Voer retentie in uren in",
|
"custom_retention_input": "Voer retentie in uren in",
|
||||||
"custom_rotation_input": "Voer rotatie in uren in",
|
"custom_rotation_input": "Voer rotatie in uren in",
|
||||||
|
|||||||
@@ -264,7 +264,7 @@
|
|||||||
"custom_ip": "Tilpasset IP",
|
"custom_ip": "Tilpasset IP",
|
||||||
"blocking_ipv4": "IPv4-blokkering",
|
"blocking_ipv4": "IPv4-blokkering",
|
||||||
"blocking_ipv6": "IPv6-blokkering",
|
"blocking_ipv6": "IPv6-blokkering",
|
||||||
"blocked_response_ttl": "Blokkert svar TTL",
|
"blocked_response_ttl": "Blokkerte svars TTL",
|
||||||
"dnscrypt": "DNSCrypt",
|
"dnscrypt": "DNSCrypt",
|
||||||
"dns_over_https": "DNS-over-HTTPS",
|
"dns_over_https": "DNS-over-HTTPS",
|
||||||
"dns_over_tls": "DNS-over-TLS",
|
"dns_over_tls": "DNS-over-TLS",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Lista de bloqueio",
|
"blocklist": "Lista de bloqueio",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Tamanho do cache",
|
"cache_size": "Tamanho do cache",
|
||||||
"cache_size_desc": "Tamanho do cache do DNS (em bytes). Para desativar o cache, deixe em branco.",
|
"cache_size_desc": "Tamanho do cache do DNS (em bytes). Para desativar o cache, defina como 0.",
|
||||||
"cache_ttl_min_override": "Sobrepor o TTL mínimo",
|
"cache_ttl_min_override": "Sobrepor o TTL mínimo",
|
||||||
"cache_ttl_max_override": "Sobrepor o TTL máximo",
|
"cache_ttl_max_override": "Sobrepor o TTL máximo",
|
||||||
"enter_cache_size": "Digite o tamanho do cache (bytes)",
|
"enter_cache_size": "Digite o tamanho do cache (bytes)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Lista de bloqueio",
|
"blocklist": "Lista de bloqueio",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Tamanho do cache",
|
"cache_size": "Tamanho do cache",
|
||||||
"cache_size_desc": "Tamanho do cache DNS (em bytes). Para desativar o cache, deixar o campo vazio.",
|
"cache_size_desc": "Tamanho do cache DNS (em bytes). Para desativar o cache, defina como 0.",
|
||||||
"cache_ttl_min_override": "Sobrepor o TTL mínimo",
|
"cache_ttl_min_override": "Sobrepor o TTL mínimo",
|
||||||
"cache_ttl_max_override": "Sobrepor o TTL máximo",
|
"cache_ttl_max_override": "Sobrepor o TTL máximo",
|
||||||
"enter_cache_size": "Digite o tamanho do cache (bytes)",
|
"enter_cache_size": "Digite o tamanho do cache (bytes)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Чёрный список",
|
"blocklist": "Чёрный список",
|
||||||
"milliseconds_abbreviation": "мс",
|
"milliseconds_abbreviation": "мс",
|
||||||
"cache_size": "Размер кеша",
|
"cache_size": "Размер кеша",
|
||||||
"cache_size_desc": "Размера кеша DNS (в байтах). Чтобы отключить кэширование, оставьте поле пустым.",
|
"cache_size_desc": "Размер кеша DNS (в байтах). Чтобы отключить кеширование, установите значение 0.",
|
||||||
"cache_ttl_min_override": "Переопределить минимальный TTL",
|
"cache_ttl_min_override": "Переопределить минимальный TTL",
|
||||||
"cache_ttl_max_override": "Переопределить максимальный TTL",
|
"cache_ttl_max_override": "Переопределить максимальный TTL",
|
||||||
"enter_cache_size": "Введите размер кеша (в байтах)",
|
"enter_cache_size": "Введите размер кеша (в байтах)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Zoznam blokovaní",
|
"blocklist": "Zoznam blokovaní",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Veľkosť cache",
|
"cache_size": "Veľkosť cache",
|
||||||
"cache_size_desc": "Veľkosť vyrovnávacej pamäte DNS (v bajtoch). Ak chcete zakázať ukladanie do vyrovnávacej pamäte, ponechajte pole prázdne.",
|
"cache_size_desc": "Veľkosť vyrovnávacej pamäte DNS (v bajtoch). Ak chcete vypnúť ukladanie do vyrovnávacej pamäte, nastavte hodnotu 0.",
|
||||||
"cache_ttl_min_override": "Prepísať minimálne TTL",
|
"cache_ttl_min_override": "Prepísať minimálne TTL",
|
||||||
"cache_ttl_max_override": "Prepísať maximálne TTL",
|
"cache_ttl_max_override": "Prepísať maximálne TTL",
|
||||||
"enter_cache_size": "Zadať veľkosť cache (v bajtoch)",
|
"enter_cache_size": "Zadať veľkosť cache (v bajtoch)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "Engel listesi",
|
"blocklist": "Engel listesi",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "Önbellek boyutu",
|
"cache_size": "Önbellek boyutu",
|
||||||
"cache_size_desc": "DNS önbellek boyutu (bayt cinsinden). Önbelleğe almayı devre dışı bırakmak için boş bırakın.",
|
"cache_size_desc": "DNS önbellek boyutu (bayt cinsinden). Önbelleği devre dışı bırakmak için 0 olarak ayarlayın.",
|
||||||
"cache_ttl_min_override": "Minimum kullanım süresini geçersiz kıl",
|
"cache_ttl_min_override": "Minimum kullanım süresini geçersiz kıl",
|
||||||
"cache_ttl_max_override": "Maksimum kullanım süresini geçersiz kıl",
|
"cache_ttl_max_override": "Maksimum kullanım süresini geçersiz kıl",
|
||||||
"enter_cache_size": "Önbellek boyutunu girin (bayt)",
|
"enter_cache_size": "Önbellek boyutunu girin (bayt)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "黑名单",
|
"blocklist": "黑名单",
|
||||||
"milliseconds_abbreviation": "毫秒",
|
"milliseconds_abbreviation": "毫秒",
|
||||||
"cache_size": "缓存大小",
|
"cache_size": "缓存大小",
|
||||||
"cache_size_desc": "DNS 缓存大小(单位:字节)。若要关闭缓存,请留空。",
|
"cache_size_desc": "DNS 缓存大小(单位:字节)。若要禁用缓存,请设置为 0。",
|
||||||
"cache_ttl_min_override": "覆盖最小 TTL 值",
|
"cache_ttl_min_override": "覆盖最小 TTL 值",
|
||||||
"cache_ttl_max_override": "覆盖最大 TTL 值",
|
"cache_ttl_max_override": "覆盖最大 TTL 值",
|
||||||
"enter_cache_size": "输入缓存大小(字节)",
|
"enter_cache_size": "输入缓存大小(字节)",
|
||||||
|
|||||||
@@ -656,7 +656,7 @@
|
|||||||
"blocklist": "封鎖清單",
|
"blocklist": "封鎖清單",
|
||||||
"milliseconds_abbreviation": "ms",
|
"milliseconds_abbreviation": "ms",
|
||||||
"cache_size": "快取大小",
|
"cache_size": "快取大小",
|
||||||
"cache_size_desc": "DNS 快取大小 (位元組)。若要停用快取,請留空。",
|
"cache_size_desc": "DNS 快取大小(位元組)。若要停用快取,請設為 0。",
|
||||||
"cache_ttl_min_override": "覆寫最小的存活時間(TTL)",
|
"cache_ttl_min_override": "覆寫最小的存活時間(TTL)",
|
||||||
"cache_ttl_max_override": "覆寫最大的存活時間(TTL)",
|
"cache_ttl_max_override": "覆寫最大的存活時間(TTL)",
|
||||||
"enter_cache_size": "輸入快取大小(位元組)",
|
"enter_cache_size": "輸入快取大小(位元組)",
|
||||||
|
|||||||
@@ -78,6 +78,7 @@ class CustomRules extends Component<CustomRulesProps> {
|
|||||||
<form onSubmit={this.handleSubmit}>
|
<form onSubmit={this.handleSubmit}>
|
||||||
<div className="text-edit-container mb-4">
|
<div className="text-edit-container mb-4">
|
||||||
<textarea
|
<textarea
|
||||||
|
data-testid="custom_rule_textarea"
|
||||||
className="form-control font-monospace text-input"
|
className="form-control font-monospace text-input"
|
||||||
value={userRules}
|
value={userRules}
|
||||||
onChange={this.handleChange}
|
onChange={this.handleChange}
|
||||||
@@ -91,6 +92,7 @@ class CustomRules extends Component<CustomRulesProps> {
|
|||||||
|
|
||||||
<div className="card-actions">
|
<div className="card-actions">
|
||||||
<button
|
<button
|
||||||
|
data-testid="apply_custom_rule"
|
||||||
className="btn btn-success btn-standard btn-large"
|
className="btn btn-success btn-standard btn-large"
|
||||||
type="submit"
|
type="submit"
|
||||||
onClick={this.handleSubmit}>
|
onClick={this.handleSubmit}>
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ const Header = () => {
|
|||||||
<div className="header__column">
|
<div className="header__column">
|
||||||
<div className="header__right">
|
<div className="header__right">
|
||||||
{!processingProfile && name && (
|
{!processingProfile && name && (
|
||||||
<a href="control/logout" className="btn btn-sm btn-outline-secondary">
|
<a href="control/logout" className="btn btn-sm btn-outline-secondary" data-testid="sign_out">
|
||||||
{t('sign_out')}
|
{t('sign_out')}
|
||||||
</a>
|
</a>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -288,7 +288,7 @@ const Row = memo(
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div style={style} className={className} onClick={onClick} role="row">
|
<div style={style} className={className} onClick={onClick} role="row" data-testid="querylog_cell">
|
||||||
<DateCell {...rowProps} />
|
<DateCell {...rowProps} />
|
||||||
|
|
||||||
<DomainCell {...rowProps} />
|
<DomainCell {...rowProps} />
|
||||||
|
|||||||
@@ -84,6 +84,7 @@ export const Form = ({ className, setIsLoading }: Props) => {
|
|||||||
}}>
|
}}>
|
||||||
<div className="field__search">
|
<div className="field__search">
|
||||||
<SearchField
|
<SearchField
|
||||||
|
data-testid="querylog_search"
|
||||||
value={searchValue}
|
value={searchValue}
|
||||||
handleChange={(val) => setValue('search', val)}
|
handleChange={(val) => setValue('search', val)}
|
||||||
onKeyDown={onEnterPress}
|
onKeyDown={onEnterPress}
|
||||||
|
|||||||
@@ -27,12 +27,14 @@ const SETTINGS = {
|
|||||||
enabled: false,
|
enabled: false,
|
||||||
title: i18next.t('use_adguard_browsing_sec'),
|
title: i18next.t('use_adguard_browsing_sec'),
|
||||||
subtitle: i18next.t('use_adguard_browsing_sec_hint'),
|
subtitle: i18next.t('use_adguard_browsing_sec_hint'),
|
||||||
|
testId: 'safebrowsing',
|
||||||
[ORDER_KEY]: 0,
|
[ORDER_KEY]: 0,
|
||||||
},
|
},
|
||||||
parental: {
|
parental: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
title: i18next.t('use_adguard_parental'),
|
title: i18next.t('use_adguard_parental'),
|
||||||
subtitle: i18next.t('use_adguard_parental_hint'),
|
subtitle: i18next.t('use_adguard_parental_hint'),
|
||||||
|
testId: 'parental',
|
||||||
[ORDER_KEY]: 1,
|
[ORDER_KEY]: 1,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@@ -90,11 +92,12 @@ class Settings extends Component<SettingsProps> {
|
|||||||
renderSettings = (settings: any) =>
|
renderSettings = (settings: any) =>
|
||||||
getObjectKeysSorted(SETTINGS, ORDER_KEY).map((key: any) => {
|
getObjectKeysSorted(SETTINGS, ORDER_KEY).map((key: any) => {
|
||||||
const setting = settings[key];
|
const setting = settings[key];
|
||||||
const { enabled, title, subtitle } = setting;
|
const { enabled, title, subtitle, testId } = setting;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div key={key} className="form__group form__group--checkbox">
|
<div key={key} className="form__group form__group--checkbox">
|
||||||
<Checkbox
|
<Checkbox
|
||||||
|
data-testid={testId}
|
||||||
value={enabled}
|
value={enabled}
|
||||||
title={title}
|
title={title}
|
||||||
subtitle={subtitle}
|
subtitle={subtitle}
|
||||||
@@ -118,6 +121,7 @@ class Settings extends Component<SettingsProps> {
|
|||||||
<>
|
<>
|
||||||
<div className="form__group form__group--checkbox">
|
<div className="form__group form__group--checkbox">
|
||||||
<Checkbox
|
<Checkbox
|
||||||
|
data-testid="safesearch"
|
||||||
value={enabled}
|
value={enabled}
|
||||||
title={i18next.t('enforce_safe_search')}
|
title={i18next.t('enforce_safe_search')}
|
||||||
subtitle={i18next.t('enforce_save_search_hint')}
|
subtitle={i18next.t('enforce_save_search_hint')}
|
||||||
|
|||||||
@@ -94,14 +94,17 @@ const Footer = () => {
|
|||||||
auto: {
|
auto: {
|
||||||
desc: t('theme_auto_desc'),
|
desc: t('theme_auto_desc'),
|
||||||
icon: '#auto',
|
icon: '#auto',
|
||||||
|
testId: 'theme_auto',
|
||||||
},
|
},
|
||||||
dark: {
|
dark: {
|
||||||
desc: t('theme_dark_desc'),
|
desc: t('theme_dark_desc'),
|
||||||
icon: '#dark',
|
icon: '#dark',
|
||||||
|
testId: 'theme_dark',
|
||||||
},
|
},
|
||||||
light: {
|
light: {
|
||||||
desc: t('theme_light_desc'),
|
desc: t('theme_light_desc'),
|
||||||
icon: '#light',
|
icon: '#light',
|
||||||
|
testId: 'theme_light',
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -113,7 +116,9 @@ const Footer = () => {
|
|||||||
type="button"
|
type="button"
|
||||||
className="btn btn-sm btn-secondary footer__theme-button"
|
className="btn btn-sm btn-secondary footer__theme-button"
|
||||||
onClick={() => onThemeChange(theme)}
|
onClick={() => onThemeChange(theme)}
|
||||||
title={content[theme].desc}>
|
title={content[theme].desc}
|
||||||
|
data-testid={content[theme].testId}
|
||||||
|
>
|
||||||
<svg className={cn('footer__theme-icon', { 'footer__theme-icon--active': currentValue === theme })}>
|
<svg className={cn('footer__theme-icon', { 'footer__theme-icon--active': currentValue === theme })}>
|
||||||
<use xlinkHref={content[theme].icon} />
|
<use xlinkHref={content[theme].icon} />
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
@@ -28,6 +28,12 @@ export default {
|
|||||||
"homepage": "https://badmojr.github.io/1Hosts/",
|
"homepage": "https://badmojr.github.io/1Hosts/",
|
||||||
"source": "https://adguardteam.github.io/HostlistsRegistry/assets/filter_24.txt"
|
"source": "https://adguardteam.github.io/HostlistsRegistry/assets/filter_24.txt"
|
||||||
},
|
},
|
||||||
|
"1hosts_pro": {
|
||||||
|
"name": "1Hosts (Pro)",
|
||||||
|
"categoryId": "general",
|
||||||
|
"homepage": "https://badmojr.github.io/1Hosts/",
|
||||||
|
"source": "https://adguardteam.github.io/HostlistsRegistry/assets/filter_64.txt"
|
||||||
|
},
|
||||||
"CHN_adrules": {
|
"CHN_adrules": {
|
||||||
"name": "CHN: AdRules DNS List",
|
"name": "CHN: AdRules DNS List",
|
||||||
"categoryId": "regional",
|
"categoryId": "regional",
|
||||||
|
|||||||
34
client/tests/e2e/control-panel.spec.ts
Normal file
34
client/tests/e2e/control-panel.spec.ts
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { ADMIN_USERNAME, ADMIN_PASSWORD } from '../constants';
|
||||||
|
|
||||||
|
test.describe('Control Panel', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('/login.html');
|
||||||
|
await page.getByTestId('username').click();
|
||||||
|
await page.getByTestId('username').fill(ADMIN_USERNAME);
|
||||||
|
await page.getByTestId('password').click();
|
||||||
|
await page.getByTestId('password').fill(ADMIN_PASSWORD);
|
||||||
|
await page.keyboard.press('Tab');
|
||||||
|
await page.getByTestId('sign_in').click();
|
||||||
|
await page.waitForURL((url) => !url.href.endsWith('/login.html'));
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should sign out successfully', async ({ page }) => {
|
||||||
|
await page.getByTestId('sign_out').click();
|
||||||
|
|
||||||
|
await page.waitForURL((url) => url.href.endsWith('/login.html'));
|
||||||
|
|
||||||
|
await expect(page.getByTestId('sign_in')).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should change theme to dark and then light', async ({ page }) => {
|
||||||
|
await page.getByTestId('theme_dark').click();
|
||||||
|
|
||||||
|
await expect(page.locator('body[data-theme="dark"]')).toBeVisible();
|
||||||
|
|
||||||
|
|
||||||
|
await page.getByTestId('theme_light').click();
|
||||||
|
|
||||||
|
await expect(page.locator('body:not([data-theme="dark"])')).toBeVisible();
|
||||||
|
});
|
||||||
|
});
|
||||||
52
client/tests/e2e/dns-settings.spec.ts
Normal file
52
client/tests/e2e/dns-settings.spec.ts
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
import { test, expect, type Page } from '@playwright/test';
|
||||||
|
import { ADMIN_USERNAME, ADMIN_PASSWORD } from '../constants';
|
||||||
|
|
||||||
|
test.describe('DNS Settings', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// Login before each test
|
||||||
|
await page.goto('/login.html');
|
||||||
|
await page.getByTestId('username').click();
|
||||||
|
await page.getByTestId('username').fill(ADMIN_USERNAME);
|
||||||
|
await page.getByTestId('password').click();
|
||||||
|
await page.getByTestId('password').fill(ADMIN_PASSWORD);
|
||||||
|
await page.keyboard.press('Tab');
|
||||||
|
await page.getByTestId('sign_in').click();
|
||||||
|
await page.waitForURL((url) => !url.href.endsWith('/login.html'));
|
||||||
|
});
|
||||||
|
|
||||||
|
const runDNSSettingsTest = async (page: Page, address: string) => {
|
||||||
|
await page.goto('/#dns');
|
||||||
|
|
||||||
|
const currentDns = await page.getByTestId('upstream_dns').inputValue();
|
||||||
|
|
||||||
|
await page.getByTestId('upstream_dns').fill(address);
|
||||||
|
await page.getByTestId('dns_upstream_test').click();
|
||||||
|
|
||||||
|
await page.waitForTimeout(2000);
|
||||||
|
|
||||||
|
await expect(page.getByTestId('upstream_dns')).toHaveValue(address);
|
||||||
|
|
||||||
|
await page.getByTestId('upstream_dns').fill(currentDns);
|
||||||
|
await page.getByTestId('dns_upstream_save').click({ force: true });
|
||||||
|
};
|
||||||
|
|
||||||
|
test('test for Default DNS', async ({ page }) => {
|
||||||
|
await runDNSSettingsTest(page, 'https://dns10.quad9.net/dns-query');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('test for Plain DNS', async ({ page }) => {
|
||||||
|
await runDNSSettingsTest(page, '94.140.14.140');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('test for DNS-over-HTTPS', async ({ page }) => {
|
||||||
|
await runDNSSettingsTest(page, 'https://unfiltered.adguard-dns.com/dns-query');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('test for DNS-over-TLS', async ({ page }) => {
|
||||||
|
await runDNSSettingsTest(page, 'tls://unfiltered.adguard-dns.com');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('test for DNS-over-QUIC', async ({ page }) => {
|
||||||
|
await runDNSSettingsTest(page, 'quic://unfiltered.adguard-dns.com');
|
||||||
|
});
|
||||||
|
});
|
||||||
73
client/tests/e2e/filtering.spec.ts
Normal file
73
client/tests/e2e/filtering.spec.ts
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
import { test, expect, type Page } from '@playwright/test';
|
||||||
|
import { execSync } from 'child_process';
|
||||||
|
import { ADMIN_USERNAME, ADMIN_PASSWORD } from '../constants';
|
||||||
|
|
||||||
|
test.describe('Filtering', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// Login before each test
|
||||||
|
await page.goto('/login.html');
|
||||||
|
await page.getByTestId('username').click();
|
||||||
|
await page.getByTestId('username').fill(ADMIN_USERNAME);
|
||||||
|
await page.getByTestId('password').click();
|
||||||
|
await page.getByTestId('password').fill(ADMIN_PASSWORD);
|
||||||
|
await page.keyboard.press('Tab');
|
||||||
|
await page.getByTestId('sign_in').click();
|
||||||
|
await page.waitForURL((url) => !url.href.endsWith('/login.html'));
|
||||||
|
});
|
||||||
|
|
||||||
|
const runTerminalCommand = (command: string) => {
|
||||||
|
try {
|
||||||
|
console.info(`Executing command: ${command}`);
|
||||||
|
|
||||||
|
const output = execSync(command, { encoding: 'utf-8', stdio: 'pipe' }).trim();
|
||||||
|
|
||||||
|
console.info('Command executed successfully.');
|
||||||
|
console.debug(`Command output:\n${output}`);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
} catch (error: any) {
|
||||||
|
console.error(`Command execution failed with error:\n${error.message}`);
|
||||||
|
throw new Error(`Failed to execute command: ${command}\nError: ${error.message}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const runCustomRuleTest = async (page: Page, domain_to_block: string) => {
|
||||||
|
await page.goto('/#custom_rules');
|
||||||
|
|
||||||
|
await page.getByTestId('custom_rule_textarea').fill(domain_to_block);
|
||||||
|
await page.getByTestId('apply_custom_rule').click();
|
||||||
|
|
||||||
|
const nslookupBlockedResult = await runTerminalCommand(`nslookup ${domain_to_block} 127.0.0.1`).toString();
|
||||||
|
|
||||||
|
console.info(`nslookup blocked CNAME result: '${nslookupBlockedResult}'`);
|
||||||
|
|
||||||
|
const currentRules = await page.getByTestId('custom_rule_textarea').inputValue();
|
||||||
|
console.debug(`Current rules before removal:\n${currentRules}`);
|
||||||
|
|
||||||
|
if (currentRules.includes(domain_to_block)) {
|
||||||
|
const updatedRules = currentRules
|
||||||
|
.split('\n')
|
||||||
|
.filter((line) => line.trim() !== domain_to_block.trim())
|
||||||
|
.join('\n');
|
||||||
|
|
||||||
|
await page.getByTestId('custom_rule_textarea').fill(updatedRules);
|
||||||
|
console.info(`Rule '${domain_to_block}' removed successfully.`);
|
||||||
|
|
||||||
|
console.info('Applying the updated filtering rules after removal.');
|
||||||
|
await page.getByTestId('apply_custom_rule').click();
|
||||||
|
|
||||||
|
await page.waitForLoadState('domcontentloaded');
|
||||||
|
|
||||||
|
console.info(`Filtering rules successfully updated after removing '${domain_to_block}'.`);
|
||||||
|
} else {
|
||||||
|
console.warn(`Rule '${domain_to_block}' not found. No changes were made.`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const nslookupUnblockedResult = await runTerminalCommand(`nslookup ${domain_to_block} 127.0.0.1`).toString();
|
||||||
|
console.info(`nslookup unblocked CNAME result: '${nslookupUnblockedResult}'`);
|
||||||
|
};
|
||||||
|
|
||||||
|
test('Test blocking rule for apple.com', async ({ page }) => {
|
||||||
|
await runCustomRuleTest(page, 'apple.com');
|
||||||
|
});
|
||||||
|
});
|
||||||
89
client/tests/e2e/general-settings.spec.ts
Normal file
89
client/tests/e2e/general-settings.spec.ts
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { execSync } from 'child_process';
|
||||||
|
import { ADMIN_USERNAME, ADMIN_PASSWORD } from '../constants';
|
||||||
|
|
||||||
|
test.describe('General Settings', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('/login.html');
|
||||||
|
await page.getByTestId('username').click();
|
||||||
|
await page.getByTestId('username').fill(ADMIN_USERNAME);
|
||||||
|
await page.getByTestId('password').click();
|
||||||
|
await page.getByTestId('password').fill(ADMIN_PASSWORD);
|
||||||
|
await page.keyboard.press('Tab');
|
||||||
|
await page.getByTestId('sign_in').click();
|
||||||
|
await page.waitForURL((url) => !url.href.endsWith('/login.html'));
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should toggle browsing security feature and verify DNS changes', async ({ page }) => {
|
||||||
|
await page.goto('/#settings');
|
||||||
|
|
||||||
|
const browsingSecurity = await page.getByTestId('safebrowsing');
|
||||||
|
const browsingSecurityLabel = await browsingSecurity.locator('xpath=following-sibling::*[1]');
|
||||||
|
|
||||||
|
const initialState = await browsingSecurity.isChecked();
|
||||||
|
|
||||||
|
if (!initialState) {
|
||||||
|
await browsingSecurityLabel.click();
|
||||||
|
await expect(browsingSecurity).toBeChecked();
|
||||||
|
}
|
||||||
|
|
||||||
|
const resultEnabled = execSync('nslookup totalvirus.com 127.0.0.1').toString();
|
||||||
|
|
||||||
|
await browsingSecurityLabel.click();
|
||||||
|
await expect(browsingSecurity).not.toBeChecked();
|
||||||
|
|
||||||
|
const resultDisabled = execSync('nslookup totalvirus.com 127.0.0.1').toString();
|
||||||
|
|
||||||
|
expect(resultEnabled).not.toEqual(resultDisabled);
|
||||||
|
|
||||||
|
if (initialState) {
|
||||||
|
await browsingSecurityLabel.click();
|
||||||
|
await expect(browsingSecurity).toBeChecked();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should toggle parental control feature and verify DNS changes', async ({ page }) => {
|
||||||
|
await page.goto('/#settings');
|
||||||
|
|
||||||
|
const parentalControl = page.getByTestId('parental');
|
||||||
|
const parentalControlLabel = await parentalControl.locator('xpath=following-sibling::*[1]');
|
||||||
|
|
||||||
|
const initialState = await parentalControl.isChecked();
|
||||||
|
|
||||||
|
if (!initialState) {
|
||||||
|
await parentalControlLabel.click();
|
||||||
|
await expect(parentalControl).toBeChecked();
|
||||||
|
}
|
||||||
|
|
||||||
|
const resultEnabled = execSync('nslookup pornhub.com 127.0.0.1').toString();
|
||||||
|
|
||||||
|
await parentalControlLabel.click();
|
||||||
|
await expect(parentalControl).not.toBeChecked();
|
||||||
|
|
||||||
|
const resultDisabled = execSync('nslookup pornhub.com 127.0.0.1').toString();
|
||||||
|
|
||||||
|
expect(resultEnabled).not.toEqual(resultDisabled);
|
||||||
|
|
||||||
|
if (initialState) {
|
||||||
|
await parentalControlLabel.click();
|
||||||
|
await expect(parentalControl).toBeChecked();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should toggle safe search feature', async ({ page }) => {
|
||||||
|
await page.goto('/#settings');
|
||||||
|
|
||||||
|
const safeSearch = page.getByTestId('safesearch');
|
||||||
|
const safeSearchLabel = await safeSearch.locator('xpath=following-sibling::*[1]');
|
||||||
|
|
||||||
|
const initialState = await safeSearch.isChecked();
|
||||||
|
|
||||||
|
await safeSearchLabel.click();
|
||||||
|
|
||||||
|
await expect(safeSearch).not.toBeChecked({ checked: initialState });
|
||||||
|
|
||||||
|
await safeSearchLabel.click();
|
||||||
|
|
||||||
|
await expect(safeSearch).toBeChecked({ checked: initialState });
|
||||||
|
});
|
||||||
|
});
|
||||||
124
client/tests/e2e/querylog.spec.ts
Normal file
124
client/tests/e2e/querylog.spec.ts
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { ADMIN_USERNAME, ADMIN_PASSWORD } from '../constants';
|
||||||
|
|
||||||
|
test.describe('QueryLog', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('/login.html');
|
||||||
|
await page.getByTestId('username').click();
|
||||||
|
await page.getByTestId('username').fill(ADMIN_USERNAME);
|
||||||
|
await page.getByTestId('password').click();
|
||||||
|
await page.getByTestId('password').fill(ADMIN_PASSWORD);
|
||||||
|
await page.keyboard.press('Tab');
|
||||||
|
await page.getByTestId('sign_in').click();
|
||||||
|
await page.waitForURL((url) => !url.href.endsWith('/login.html'));
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Search of queryLog should work correctly', async ({ page }) => {
|
||||||
|
await page.route('/control/querylog', async (route) => {
|
||||||
|
await route.fulfill({
|
||||||
|
status: 200,
|
||||||
|
contentType: 'application/json',
|
||||||
|
body: JSON.stringify(
|
||||||
|
{
|
||||||
|
"data": [
|
||||||
|
{
|
||||||
|
"answer": [
|
||||||
|
{
|
||||||
|
"type": "A",
|
||||||
|
"value": "77.88.44.242",
|
||||||
|
"ttl": 294
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "A",
|
||||||
|
"value": "5.255.255.242",
|
||||||
|
"ttl": 294
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "A",
|
||||||
|
"value": "77.88.55.242",
|
||||||
|
"ttl": 294
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"answer_dnssec": false,
|
||||||
|
"cached": false,
|
||||||
|
"client": "127.0.0.1",
|
||||||
|
"client_info": {
|
||||||
|
"whois": {},
|
||||||
|
"name": "localhost",
|
||||||
|
"disallowed_rule": "127.0.0.1",
|
||||||
|
"disallowed": false
|
||||||
|
},
|
||||||
|
"client_proto": "",
|
||||||
|
"elapsedMs": "78.163167",
|
||||||
|
"question": {
|
||||||
|
"class": "IN",
|
||||||
|
"name": "ya.ru",
|
||||||
|
"type": "A"
|
||||||
|
},
|
||||||
|
"reason": "NotFilteredNotFound",
|
||||||
|
"rules": [],
|
||||||
|
"status": "NOERROR",
|
||||||
|
"time": "2024-07-17T16:02:37.500662+02:00",
|
||||||
|
"upstream": "https://dns10.quad9.net:443/dns-query"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"answer": [
|
||||||
|
{
|
||||||
|
"type": "A",
|
||||||
|
"value": "77.88.55.242",
|
||||||
|
"ttl": 351
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "A",
|
||||||
|
"value": "77.88.44.242",
|
||||||
|
"ttl": 351
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "A",
|
||||||
|
"value": "5.255.255.242",
|
||||||
|
"ttl": 351
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"answer_dnssec": false,
|
||||||
|
"cached": false,
|
||||||
|
"client": "127.0.0.1",
|
||||||
|
"client_info": {
|
||||||
|
"whois": {},
|
||||||
|
"name": "localhost",
|
||||||
|
"disallowed_rule": "127.0.0.1",
|
||||||
|
"disallowed": false
|
||||||
|
},
|
||||||
|
"client_proto": "",
|
||||||
|
"elapsedMs": "5051.070708",
|
||||||
|
"question": {
|
||||||
|
"class": "IN",
|
||||||
|
"name": "ya.ru",
|
||||||
|
"type": "A"
|
||||||
|
},
|
||||||
|
"reason": "NotFilteredNotFound",
|
||||||
|
"rules": [],
|
||||||
|
"status": "NOERROR",
|
||||||
|
"time": "2024-07-17T16:02:37.4983+02:00",
|
||||||
|
"upstream": "https://dns10.quad9.net:443/dns-query"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"oldest": "2024-07-17T16:02:37.4983+02:00"
|
||||||
|
}
|
||||||
|
),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto('/#logs');
|
||||||
|
|
||||||
|
await page.getByTestId('querylog_search').fill('127.0.0.1');
|
||||||
|
|
||||||
|
const [request] = await Promise.all([
|
||||||
|
page.waitForRequest((req) => req.url().includes('/control/querylog')),
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (request) {
|
||||||
|
expect(request.url()).toContain('search=127.0.0.1');
|
||||||
|
expect(await page.getByTestId('querylog_cell').first().isVisible()).toBe(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1,12 +1,12 @@
|
|||||||
# A docker file for scripts/make/build-docker.sh.
|
# A docker file for scripts/make/build-docker.sh.
|
||||||
|
|
||||||
FROM alpine:3.18
|
FROM alpine:3.21
|
||||||
|
|
||||||
ARG BUILD_DATE
|
ARG BUILD_DATE
|
||||||
ARG VERSION
|
ARG VERSION
|
||||||
ARG VCS_REF
|
ARG VCS_REF
|
||||||
|
|
||||||
LABEL\
|
LABEL \
|
||||||
maintainer="AdGuard Team <devteam@adguard.com>" \
|
maintainer="AdGuard Team <devteam@adguard.com>" \
|
||||||
org.opencontainers.image.authors="AdGuard Team <devteam@adguard.com>" \
|
org.opencontainers.image.authors="AdGuard Team <devteam@adguard.com>" \
|
||||||
org.opencontainers.image.created=$BUILD_DATE \
|
org.opencontainers.image.created=$BUILD_DATE \
|
||||||
@@ -30,8 +30,8 @@ ARG TARGETARCH
|
|||||||
ARG TARGETOS
|
ARG TARGETOS
|
||||||
ARG TARGETVARIANT
|
ARG TARGETVARIANT
|
||||||
|
|
||||||
COPY --chown=nobody:nogroup\
|
COPY --chown=nobody:nogroup \
|
||||||
./${DIST_DIR}/docker/AdGuardHome_${TARGETOS}_${TARGETARCH}_${TARGETVARIANT}\
|
./${DIST_DIR}/docker/AdGuardHome_${TARGETOS}_${TARGETARCH}_${TARGETVARIANT} \
|
||||||
/opt/adguardhome/AdGuardHome
|
/opt/adguardhome/AdGuardHome
|
||||||
|
|
||||||
RUN setcap 'cap_net_bind_service=+eip' /opt/adguardhome/AdGuardHome
|
RUN setcap 'cap_net_bind_service=+eip' /opt/adguardhome/AdGuardHome
|
||||||
@@ -45,8 +45,15 @@ RUN setcap 'cap_net_bind_service=+eip' /opt/adguardhome/AdGuardHome
|
|||||||
# 3000 : TCP, UDP : HTTP(S) (alt, incl. HTTP/3)
|
# 3000 : TCP, UDP : HTTP(S) (alt, incl. HTTP/3)
|
||||||
# 5443 : TCP, UDP : DNSCrypt (alt)
|
# 5443 : TCP, UDP : DNSCrypt (alt)
|
||||||
# 6060 : TCP : HTTP (pprof)
|
# 6060 : TCP : HTTP (pprof)
|
||||||
EXPOSE 53/tcp 53/udp 67/udp 68/udp 80/tcp 443/tcp 443/udp 853/tcp\
|
EXPOSE 53/tcp 53/udp \
|
||||||
853/udp 3000/tcp 3000/udp 5443/tcp 5443/udp 6060/tcp
|
67/udp \
|
||||||
|
68/udp \
|
||||||
|
80/tcp \
|
||||||
|
443/tcp 443/udp \
|
||||||
|
853/tcp 853/udp \
|
||||||
|
3000/tcp 3000/udp \
|
||||||
|
5443/tcp 5443/udp \
|
||||||
|
6060/tcp
|
||||||
|
|
||||||
WORKDIR /opt/adguardhome/work
|
WORKDIR /opt/adguardhome/work
|
||||||
|
|
||||||
|
|||||||
26
go.mod
26
go.mod
@@ -3,8 +3,8 @@ module github.com/AdguardTeam/AdGuardHome
|
|||||||
go 1.24.2
|
go 1.24.2
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/AdguardTeam/dnsproxy v0.75.2
|
github.com/AdguardTeam/dnsproxy v0.75.4
|
||||||
github.com/AdguardTeam/golibs v0.32.7
|
github.com/AdguardTeam/golibs v0.32.8
|
||||||
github.com/AdguardTeam/urlfilter v0.20.0
|
github.com/AdguardTeam/urlfilter v0.20.0
|
||||||
github.com/NYTimes/gziphandler v1.1.1
|
github.com/NYTimes/gziphandler v1.1.1
|
||||||
github.com/ameshkov/dnscrypt/v2 v2.4.0
|
github.com/ameshkov/dnscrypt/v2 v2.4.0
|
||||||
@@ -34,7 +34,7 @@ require (
|
|||||||
github.com/ti-mo/netfilter v0.5.2
|
github.com/ti-mo/netfilter v0.5.2
|
||||||
go.etcd.io/bbolt v1.4.0
|
go.etcd.io/bbolt v1.4.0
|
||||||
golang.org/x/crypto v0.37.0
|
golang.org/x/crypto v0.37.0
|
||||||
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394
|
golang.org/x/exp v0.0.0-20250408133849-7e4ce0ab07d0
|
||||||
golang.org/x/net v0.39.0
|
golang.org/x/net v0.39.0
|
||||||
golang.org/x/sys v0.32.0
|
golang.org/x/sys v0.32.0
|
||||||
gopkg.in/natefinch/lumberjack.v2 v2.2.1
|
gopkg.in/natefinch/lumberjack.v2 v2.2.1
|
||||||
@@ -43,12 +43,12 @@ require (
|
|||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
cloud.google.com/go v0.120.0 // indirect
|
cloud.google.com/go v0.120.1 // indirect
|
||||||
cloud.google.com/go/ai v0.10.1 // indirect
|
cloud.google.com/go/ai v0.10.2 // indirect
|
||||||
cloud.google.com/go/auth v0.15.0 // indirect
|
cloud.google.com/go/auth v0.16.0 // indirect
|
||||||
cloud.google.com/go/auth/oauth2adapt v0.2.8 // indirect
|
cloud.google.com/go/auth/oauth2adapt v0.2.8 // indirect
|
||||||
cloud.google.com/go/compute/metadata v0.6.0 // indirect
|
cloud.google.com/go/compute/metadata v0.6.0 // indirect
|
||||||
cloud.google.com/go/longrunning v0.6.6 // indirect
|
cloud.google.com/go/longrunning v0.6.7 // indirect
|
||||||
github.com/BurntSushi/toml v1.5.0 // indirect
|
github.com/BurntSushi/toml v1.5.0 // indirect
|
||||||
github.com/ameshkov/dnsstamps v1.0.3 // indirect
|
github.com/ameshkov/dnsstamps v1.0.3 // indirect
|
||||||
github.com/beefsack/go-rate v0.0.0-20220214233405-116f4ca011a0 // indirect
|
github.com/beefsack/go-rate v0.0.0-20220214233405-116f4ca011a0 // indirect
|
||||||
@@ -90,25 +90,25 @@ require (
|
|||||||
go.opentelemetry.io/otel/trace v1.35.0 // indirect
|
go.opentelemetry.io/otel/trace v1.35.0 // indirect
|
||||||
go.uber.org/automaxprocs v1.6.0 // indirect
|
go.uber.org/automaxprocs v1.6.0 // indirect
|
||||||
go.uber.org/mock v0.5.1 // indirect
|
go.uber.org/mock v0.5.1 // indirect
|
||||||
golang.org/x/exp/typeparams v0.0.0-20250305212735-054e65f0b394 // indirect
|
golang.org/x/exp/typeparams v0.0.0-20250408133849-7e4ce0ab07d0 // indirect
|
||||||
golang.org/x/mod v0.24.0 // indirect
|
golang.org/x/mod v0.24.0 // indirect
|
||||||
golang.org/x/oauth2 v0.29.0 // indirect
|
golang.org/x/oauth2 v0.29.0 // indirect
|
||||||
golang.org/x/sync v0.13.0 // indirect
|
golang.org/x/sync v0.13.0 // indirect
|
||||||
golang.org/x/telemetry v0.0.0-20250406004356-f593adaf3fc1 // indirect
|
golang.org/x/telemetry v0.0.0-20250417124945-06ef541f3fa3 // indirect
|
||||||
golang.org/x/term v0.31.0 // indirect
|
golang.org/x/term v0.31.0 // indirect
|
||||||
golang.org/x/text v0.24.0 // indirect
|
golang.org/x/text v0.24.0 // indirect
|
||||||
golang.org/x/time v0.11.0 // indirect
|
golang.org/x/time v0.11.0 // indirect
|
||||||
golang.org/x/tools v0.32.0 // indirect
|
golang.org/x/tools v0.32.0 // indirect
|
||||||
golang.org/x/vuln v1.1.4 // indirect
|
golang.org/x/vuln v1.1.4 // indirect
|
||||||
gonum.org/v1/gonum v0.16.0 // indirect
|
gonum.org/v1/gonum v0.16.0 // indirect
|
||||||
google.golang.org/api v0.228.0 // indirect
|
google.golang.org/api v0.229.0 // indirect
|
||||||
google.golang.org/genproto/googleapis/api v0.0.0-20250407143221-ac9807e6c755 // indirect
|
google.golang.org/genproto/googleapis/api v0.0.0-20250414145226-207652e42e2e // indirect
|
||||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20250407143221-ac9807e6c755 // indirect
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20250414145226-207652e42e2e // indirect
|
||||||
google.golang.org/grpc v1.71.1 // indirect
|
google.golang.org/grpc v1.71.1 // indirect
|
||||||
google.golang.org/protobuf v1.36.6 // indirect
|
google.golang.org/protobuf v1.36.6 // indirect
|
||||||
honnef.co/go/tools v0.6.1 // indirect
|
honnef.co/go/tools v0.6.1 // indirect
|
||||||
mvdan.cc/editorconfig v0.3.0 // indirect
|
mvdan.cc/editorconfig v0.3.0 // indirect
|
||||||
mvdan.cc/gofumpt v0.7.0 // indirect
|
mvdan.cc/gofumpt v0.8.0 // indirect
|
||||||
mvdan.cc/sh/v3 v3.11.0 // indirect
|
mvdan.cc/sh/v3 v3.11.0 // indirect
|
||||||
mvdan.cc/unparam v0.0.0-20250301125049-0df0534333a4 // indirect
|
mvdan.cc/unparam v0.0.0-20250301125049-0df0534333a4 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
52
go.sum
52
go.sum
@@ -1,19 +1,19 @@
|
|||||||
cloud.google.com/go v0.120.0 h1:wc6bgG9DHyKqF5/vQvX1CiZrtHnxJjBlKUyF9nP6meA=
|
cloud.google.com/go v0.120.1 h1:Z+5V7yd383+9617XDCyszmK5E4wJRJL+tquMfDj9hLM=
|
||||||
cloud.google.com/go v0.120.0/go.mod h1:/beW32s8/pGRuj4IILWQNd4uuebeT4dkOhKmkfit64Q=
|
cloud.google.com/go v0.120.1/go.mod h1:56Vs7sf/i2jYM6ZL9NYlC82r04PThNcPS5YgFmb0rp8=
|
||||||
cloud.google.com/go/ai v0.10.1 h1:EU93KqYmMeOKgaBXAz2DshH2C/BzAT1P+iJORksLIic=
|
cloud.google.com/go/ai v0.10.2 h1:5NHzmZlRs+3kvlsVdjT0cTnLrjQdROJ/8VOljVfs+8o=
|
||||||
cloud.google.com/go/ai v0.10.1/go.mod h1:sWWHZvmJ83BjuxAQtYEiA0SFTpijtbH+SXWFO14ri5A=
|
cloud.google.com/go/ai v0.10.2/go.mod h1:xZuZuE9d3RgsR132meCnPadiU9XV0qXjpLr+P4J46eE=
|
||||||
cloud.google.com/go/auth v0.15.0 h1:Ly0u4aA5vG/fsSsxu98qCQBemXtAtJf+95z9HK+cxps=
|
cloud.google.com/go/auth v0.16.0 h1:Pd8P1s9WkcrBE2n/PhAwKsdrR35V3Sg2II9B+ndM3CU=
|
||||||
cloud.google.com/go/auth v0.15.0/go.mod h1:WJDGqZ1o9E9wKIL+IwStfyn/+s59zl4Bi+1KQNVXLZ8=
|
cloud.google.com/go/auth v0.16.0/go.mod h1:1howDHJ5IETh/LwYs3ZxvlkXF48aSqqJUM+5o02dNOI=
|
||||||
cloud.google.com/go/auth/oauth2adapt v0.2.8 h1:keo8NaayQZ6wimpNSmW5OPc283g65QNIiLpZnkHRbnc=
|
cloud.google.com/go/auth/oauth2adapt v0.2.8 h1:keo8NaayQZ6wimpNSmW5OPc283g65QNIiLpZnkHRbnc=
|
||||||
cloud.google.com/go/auth/oauth2adapt v0.2.8/go.mod h1:XQ9y31RkqZCcwJWNSx2Xvric3RrU88hAYYbjDWYDL+c=
|
cloud.google.com/go/auth/oauth2adapt v0.2.8/go.mod h1:XQ9y31RkqZCcwJWNSx2Xvric3RrU88hAYYbjDWYDL+c=
|
||||||
cloud.google.com/go/compute/metadata v0.6.0 h1:A6hENjEsCDtC1k8byVsgwvVcioamEHvZ4j01OwKxG9I=
|
cloud.google.com/go/compute/metadata v0.6.0 h1:A6hENjEsCDtC1k8byVsgwvVcioamEHvZ4j01OwKxG9I=
|
||||||
cloud.google.com/go/compute/metadata v0.6.0/go.mod h1:FjyFAW1MW0C203CEOMDTu3Dk1FlqW3Rga40jzHL4hfg=
|
cloud.google.com/go/compute/metadata v0.6.0/go.mod h1:FjyFAW1MW0C203CEOMDTu3Dk1FlqW3Rga40jzHL4hfg=
|
||||||
cloud.google.com/go/longrunning v0.6.6 h1:XJNDo5MUfMM05xK3ewpbSdmt7R2Zw+aQEMbdQR65Rbw=
|
cloud.google.com/go/longrunning v0.6.7 h1:IGtfDWHhQCgCjwQjV9iiLnUta9LBCo8R9QmAFsS/PrE=
|
||||||
cloud.google.com/go/longrunning v0.6.6/go.mod h1:hyeGJUrPHcx0u2Uu1UFSoYZLn4lkMrccJig0t4FI7yw=
|
cloud.google.com/go/longrunning v0.6.7/go.mod h1:EAFV3IZAKmM56TyiE6VAP3VoTzhZzySwI/YI1s/nRsY=
|
||||||
github.com/AdguardTeam/dnsproxy v0.75.2 h1:bciOkzQh/GG8vcZGdFn6+rS3pu+2Npt9tbA4bNA/rsc=
|
github.com/AdguardTeam/dnsproxy v0.75.4 h1:hTnHh9HoTYKKhKqePpIxCzfecl7dAXykZTw2gcj0I5U=
|
||||||
github.com/AdguardTeam/dnsproxy v0.75.2/go.mod h1:U/ouLftmXMIrkTAf8JepqbPuoQzsbXJo0Vxxn+LAdgA=
|
github.com/AdguardTeam/dnsproxy v0.75.4/go.mod h1:50OyTHao+uQzUJiXay08hgfvWQ3o2Q2WV99W8u8ypDE=
|
||||||
github.com/AdguardTeam/golibs v0.32.7 h1:3dmGlAVgmvquCCwHsvEl58KKcRAK3z1UnjMnwSIeDH4=
|
github.com/AdguardTeam/golibs v0.32.8 h1:O3mc3kYcPkW3kbmd+gqzFNgUka13a+iBgFLThwOYSQE=
|
||||||
github.com/AdguardTeam/golibs v0.32.7/go.mod h1:bE8KV1zqTzgZjmjFyBJ9f9O5DEKO717r7e57j1HclJA=
|
github.com/AdguardTeam/golibs v0.32.8/go.mod h1:McV1QFFlKLElKa306V4OL/T2kr7564PhsayfvTWYBVs=
|
||||||
github.com/AdguardTeam/urlfilter v0.20.0 h1:X32qiuVCVd8WDYCEsbdZKfXMzwdVqrdulamtUi4rmzs=
|
github.com/AdguardTeam/urlfilter v0.20.0 h1:X32qiuVCVd8WDYCEsbdZKfXMzwdVqrdulamtUi4rmzs=
|
||||||
github.com/AdguardTeam/urlfilter v0.20.0/go.mod h1:gjrywLTxfJh6JOkwi9SU+frhP7kVVEZ5exFGkR99qpk=
|
github.com/AdguardTeam/urlfilter v0.20.0/go.mod h1:gjrywLTxfJh6JOkwi9SU+frhP7kVVEZ5exFGkR99qpk=
|
||||||
github.com/BurntSushi/toml v1.5.0 h1:W5quZX/G/csjUnuI8SUYlsHs9M38FC7znL0lIO+DvMg=
|
github.com/BurntSushi/toml v1.5.0 h1:W5quZX/G/csjUnuI8SUYlsHs9M38FC7znL0lIO+DvMg=
|
||||||
@@ -205,10 +205,10 @@ golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACk
|
|||||||
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||||
golang.org/x/crypto v0.37.0 h1:kJNSjF/Xp7kU0iB2Z+9viTPMW4EqqsrywMXLJOOsXSE=
|
golang.org/x/crypto v0.37.0 h1:kJNSjF/Xp7kU0iB2Z+9viTPMW4EqqsrywMXLJOOsXSE=
|
||||||
golang.org/x/crypto v0.37.0/go.mod h1:vg+k43peMZ0pUMhYmVAWysMK35e6ioLh3wB8ZCAfbVc=
|
golang.org/x/crypto v0.37.0/go.mod h1:vg+k43peMZ0pUMhYmVAWysMK35e6ioLh3wB8ZCAfbVc=
|
||||||
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394 h1:nDVHiLt8aIbd/VzvPWN6kSOPE7+F/fNFDSXLVYkE/Iw=
|
golang.org/x/exp v0.0.0-20250408133849-7e4ce0ab07d0 h1:R84qjqJb5nVJMxqWYb3np9L5ZsaDtB+a39EqjV0JSUM=
|
||||||
golang.org/x/exp v0.0.0-20250305212735-054e65f0b394/go.mod h1:sIifuuw/Yco/y6yb6+bDNfyeQ/MdPUy/hKEMYQV17cM=
|
golang.org/x/exp v0.0.0-20250408133849-7e4ce0ab07d0/go.mod h1:S9Xr4PYopiDyqSyp5NjCrhFrqg6A5zA2E/iPHPhqnS8=
|
||||||
golang.org/x/exp/typeparams v0.0.0-20250305212735-054e65f0b394 h1:VI4qDpTkfFaCXEPrbojidLgVQhj2x4nzTccG0hjaLlU=
|
golang.org/x/exp/typeparams v0.0.0-20250408133849-7e4ce0ab07d0 h1:oMe07YcizemJ09rs2kRkFYAp0pt4e1lYLwPWiEGMpXE=
|
||||||
golang.org/x/exp/typeparams v0.0.0-20250305212735-054e65f0b394/go.mod h1:LKZHyeOpPuZcMgxeHjJp4p5yvxrCX1xDvH10zYHhjjQ=
|
golang.org/x/exp/typeparams v0.0.0-20250408133849-7e4ce0ab07d0/go.mod h1:LKZHyeOpPuZcMgxeHjJp4p5yvxrCX1xDvH10zYHhjjQ=
|
||||||
golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY=
|
golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY=
|
||||||
golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
|
golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
|
||||||
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||||
@@ -243,8 +243,8 @@ golang.org/x/sys v0.0.0-20220209214540-3681064d5158/go.mod h1:oPkhp1MJrh7nUepCBc
|
|||||||
golang.org/x/sys v0.4.1-0.20230131160137-e7d7f63158de/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.4.1-0.20230131160137-e7d7f63158de/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.32.0 h1:s77OFDvIQeibCmezSnk/q6iAfkdiQaJi4VzroCFrN20=
|
golang.org/x/sys v0.32.0 h1:s77OFDvIQeibCmezSnk/q6iAfkdiQaJi4VzroCFrN20=
|
||||||
golang.org/x/sys v0.32.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
|
golang.org/x/sys v0.32.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
|
||||||
golang.org/x/telemetry v0.0.0-20250406004356-f593adaf3fc1 h1:LxyDqgHX2VuimV2UQSNFpQxz+NRUUsh8ulNcP3WvNG0=
|
golang.org/x/telemetry v0.0.0-20250417124945-06ef541f3fa3 h1:RXY2+rSHXvxO2Y+gKrPjYVaEoGOqh3VEXFhnWAt1Irg=
|
||||||
golang.org/x/telemetry v0.0.0-20250406004356-f593adaf3fc1/go.mod h1:RoaXAWDwS90j6FxVKwJdBV+0HCU+llrKUGgJaxiKl6M=
|
golang.org/x/telemetry v0.0.0-20250417124945-06ef541f3fa3/go.mod h1:RoaXAWDwS90j6FxVKwJdBV+0HCU+llrKUGgJaxiKl6M=
|
||||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
golang.org/x/term v0.31.0 h1:erwDkOK1Msy6offm1mOgvspSkslFnIGsFnxOKoufg3o=
|
golang.org/x/term v0.31.0 h1:erwDkOK1Msy6offm1mOgvspSkslFnIGsFnxOKoufg3o=
|
||||||
golang.org/x/term v0.31.0/go.mod h1:R4BeIy7D95HzImkxGkTW1UQTtP54tio2RyHz7PwK0aw=
|
golang.org/x/term v0.31.0/go.mod h1:R4BeIy7D95HzImkxGkTW1UQTtP54tio2RyHz7PwK0aw=
|
||||||
@@ -268,12 +268,12 @@ golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8T
|
|||||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
gonum.org/v1/gonum v0.16.0 h1:5+ul4Swaf3ESvrOnidPp4GZbzf0mxVQpDCYUQE7OJfk=
|
gonum.org/v1/gonum v0.16.0 h1:5+ul4Swaf3ESvrOnidPp4GZbzf0mxVQpDCYUQE7OJfk=
|
||||||
gonum.org/v1/gonum v0.16.0/go.mod h1:fef3am4MQ93R2HHpKnLk4/Tbh/s0+wqD5nfa6Pnwy4E=
|
gonum.org/v1/gonum v0.16.0/go.mod h1:fef3am4MQ93R2HHpKnLk4/Tbh/s0+wqD5nfa6Pnwy4E=
|
||||||
google.golang.org/api v0.228.0 h1:X2DJ/uoWGnY5obVjewbp8icSL5U4FzuCfy9OjbLSnLs=
|
google.golang.org/api v0.229.0 h1:p98ymMtqeJ5i3lIBMj5MpR9kzIIgzpHHh8vQ+vgAzx8=
|
||||||
google.golang.org/api v0.228.0/go.mod h1:wNvRS1Pbe8r4+IfBIniV8fwCpGwTrYa+kMUDiC5z5a4=
|
google.golang.org/api v0.229.0/go.mod h1:wyDfmq5g1wYJWn29O22FDWN48P7Xcz0xz+LBpptYvB0=
|
||||||
google.golang.org/genproto/googleapis/api v0.0.0-20250407143221-ac9807e6c755 h1:AMLTAunltONNuzWgVPZXrjLWtXpsG6A3yLLPEoJ/IjU=
|
google.golang.org/genproto/googleapis/api v0.0.0-20250414145226-207652e42e2e h1:UdXH7Kzbj+Vzastr5nVfccbmFsmYNygVLSPk1pEfDoY=
|
||||||
google.golang.org/genproto/googleapis/api v0.0.0-20250407143221-ac9807e6c755/go.mod h1:2R6XrVC8Oc08GlNh8ujEpc7HkLiEZ16QeY7FxIs20ac=
|
google.golang.org/genproto/googleapis/api v0.0.0-20250414145226-207652e42e2e/go.mod h1:085qFyf2+XaZlRdCgKNCIZ3afY2p4HHZdoIRpId8F4A=
|
||||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20250407143221-ac9807e6c755 h1:TwXJCGVREgQ/cl18iY0Z4wJCTL/GmW+Um2oSwZiZPnc=
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20250414145226-207652e42e2e h1:ztQaXfzEXTmCBvbtWYRhJxW+0iJcz2qXfd38/e9l7bA=
|
||||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20250407143221-ac9807e6c755/go.mod h1:qQ0YXyHHx3XkvlzUtpXDkS29lDSafHMZBAZDc03LQ3A=
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20250414145226-207652e42e2e/go.mod h1:qQ0YXyHHx3XkvlzUtpXDkS29lDSafHMZBAZDc03LQ3A=
|
||||||
google.golang.org/grpc v1.71.1 h1:ffsFWr7ygTUscGPI0KKK6TLrGz0476KUvvsbqWK0rPI=
|
google.golang.org/grpc v1.71.1 h1:ffsFWr7ygTUscGPI0KKK6TLrGz0476KUvvsbqWK0rPI=
|
||||||
google.golang.org/grpc v1.71.1/go.mod h1:H0GRtasmQOh9LkFoCPDu3ZrwUtD1YGE+b2vYBYd/8Ec=
|
google.golang.org/grpc v1.71.1/go.mod h1:H0GRtasmQOh9LkFoCPDu3ZrwUtD1YGE+b2vYBYd/8Ec=
|
||||||
google.golang.org/protobuf v1.36.6 h1:z1NpPI8ku2WgiWnf+t9wTPsn6eP1L7ksHUlkfLvd9xY=
|
google.golang.org/protobuf v1.36.6 h1:z1NpPI8ku2WgiWnf+t9wTPsn6eP1L7ksHUlkfLvd9xY=
|
||||||
@@ -292,8 +292,8 @@ howett.net/plist v1.0.1 h1:37GdZ8tP09Q35o9ych3ehygcsL+HqKSwzctveSlarvM=
|
|||||||
howett.net/plist v1.0.1/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g=
|
howett.net/plist v1.0.1/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g=
|
||||||
mvdan.cc/editorconfig v0.3.0 h1:D1D2wLYEYGpawWT5SpM5pRivgEgXjtEXwC9MWhEY0gQ=
|
mvdan.cc/editorconfig v0.3.0 h1:D1D2wLYEYGpawWT5SpM5pRivgEgXjtEXwC9MWhEY0gQ=
|
||||||
mvdan.cc/editorconfig v0.3.0/go.mod h1:NcJHuDtNOTEJ6251indKiWuzK6+VcrMuLzGMLKBFupQ=
|
mvdan.cc/editorconfig v0.3.0/go.mod h1:NcJHuDtNOTEJ6251indKiWuzK6+VcrMuLzGMLKBFupQ=
|
||||||
mvdan.cc/gofumpt v0.7.0 h1:bg91ttqXmi9y2xawvkuMXyvAA/1ZGJqYAEGjXuP0JXU=
|
mvdan.cc/gofumpt v0.8.0 h1:nZUCeC2ViFaerTcYKstMmfysj6uhQrA2vJe+2vwGU6k=
|
||||||
mvdan.cc/gofumpt v0.7.0/go.mod h1:txVFJy/Sc/mvaycET54pV8SW8gWxTlUuGHVEcncmNUo=
|
mvdan.cc/gofumpt v0.8.0/go.mod h1:vEYnSzyGPmjvFkqJWtXkh79UwPWP9/HMxQdGEXZHjpg=
|
||||||
mvdan.cc/sh/v3 v3.11.0 h1:q5h+XMDRfUGUedCqFFsjoFjrhwf2Mvtt1rkMvVz0blw=
|
mvdan.cc/sh/v3 v3.11.0 h1:q5h+XMDRfUGUedCqFFsjoFjrhwf2Mvtt1rkMvVz0blw=
|
||||||
mvdan.cc/sh/v3 v3.11.0/go.mod h1:LRM+1NjoYCzuq/WZ6y44x14YNAI0NK7FLPeQSaFagGg=
|
mvdan.cc/sh/v3 v3.11.0/go.mod h1:LRM+1NjoYCzuq/WZ6y44x14YNAI0NK7FLPeQSaFagGg=
|
||||||
mvdan.cc/unparam v0.0.0-20250301125049-0df0534333a4 h1:WjUu4yQoT5BHT1w8Zu56SP8367OuBV5jvo+4Ulppyf8=
|
mvdan.cc/unparam v0.0.0-20250301125049-0df0534333a4 h1:WjUu4yQoT5BHT1w8Zu56SP8367OuBV5jvo+4Ulppyf8=
|
||||||
|
|||||||
@@ -10,7 +10,8 @@ import (
|
|||||||
// Login is the type for web user logins.
|
// Login is the type for web user logins.
|
||||||
type Login string
|
type Login string
|
||||||
|
|
||||||
// NewLogin returns a web user login.
|
// NewLogin returns a web user login. The length of s must not be greater than
|
||||||
|
// [math.MaxUint16].
|
||||||
//
|
//
|
||||||
// TODO(s.chzhen): Add more constraints as needed.
|
// TODO(s.chzhen): Add more constraints as needed.
|
||||||
func NewLogin(s string) (l Login, err error) {
|
func NewLogin(s string) (l Login, err error) {
|
||||||
|
|||||||
35
internal/aghuser/session.go
Normal file
35
internal/aghuser/session.go
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
package aghuser
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/rand"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SessionToken is the type for the web user session token.
|
||||||
|
type SessionToken [16]byte
|
||||||
|
|
||||||
|
// NewSessionToken returns a cryptographically secure randomly generated web
|
||||||
|
// user session token. If an error occurs during random generation, it will
|
||||||
|
// cause the program to crash.
|
||||||
|
func NewSessionToken() (t SessionToken) {
|
||||||
|
_, _ = rand.Read(t[:])
|
||||||
|
|
||||||
|
return t
|
||||||
|
}
|
||||||
|
|
||||||
|
// Session represents a web user session.
|
||||||
|
type Session struct {
|
||||||
|
// Expire indicates when the session will expire.
|
||||||
|
Expire time.Time
|
||||||
|
|
||||||
|
// UserLogin is the login of the web user associated with the session.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Remove this field and associate the user by UserID.
|
||||||
|
UserLogin Login
|
||||||
|
|
||||||
|
// Token is the session token.
|
||||||
|
Token SessionToken
|
||||||
|
|
||||||
|
// UserID is the identifier of the web user associated with the session.
|
||||||
|
UserID UserID
|
||||||
|
}
|
||||||
449
internal/aghuser/sessionstorage.go
Normal file
449
internal/aghuser/sessionstorage.go
Normal file
@@ -0,0 +1,449 @@
|
|||||||
|
package aghuser
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
|
"log/slog"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/AdguardTeam/AdGuardHome/internal/aghos"
|
||||||
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
|
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
||||||
|
"github.com/AdguardTeam/golibs/timeutil"
|
||||||
|
"go.etcd.io/bbolt"
|
||||||
|
berrors "go.etcd.io/bbolt/errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SessionStorage is an interface that defines methods for handling web user
|
||||||
|
// sessions. All methods must be safe for concurrent use.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Add DeleteAll method.
|
||||||
|
type SessionStorage interface {
|
||||||
|
// New creates a new session for the web user.
|
||||||
|
New(ctx context.Context, u *User) (s *Session, err error)
|
||||||
|
|
||||||
|
// FindByToken returns the stored session for the web user based on the session
|
||||||
|
// token.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Consider function signature change to reflect the
|
||||||
|
// in-memory implementation, as it currently always returns nil for error.
|
||||||
|
FindByToken(ctx context.Context, t SessionToken) (s *Session, err error)
|
||||||
|
|
||||||
|
// DeleteByToken removes a stored web user session by the provided token.
|
||||||
|
DeleteByToken(ctx context.Context, t SessionToken) (err error)
|
||||||
|
|
||||||
|
// Close releases the web user sessions database resources.
|
||||||
|
Close() (err error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultSessionStorageConfig represents the web user session storage
|
||||||
|
// configuration structure.
|
||||||
|
type DefaultSessionStorageConfig struct {
|
||||||
|
// Logger is used for logging the operation of the session storage. It must
|
||||||
|
// not be nil.
|
||||||
|
Logger *slog.Logger
|
||||||
|
|
||||||
|
// Clock is used to get the current time. It must not be nil.
|
||||||
|
Clock timeutil.Clock
|
||||||
|
|
||||||
|
// UserDB contains the web user information such as ID, login, and password.
|
||||||
|
// It must not be nil.
|
||||||
|
UserDB DB
|
||||||
|
|
||||||
|
// DBPath is the path to the database file where session data is stored. It
|
||||||
|
// must not be empty.
|
||||||
|
DBPath string
|
||||||
|
|
||||||
|
// SessionTTL is the default Time-To-Live duration for web user sessions.
|
||||||
|
// It specifies how long a session should last and is a required field.
|
||||||
|
SessionTTL time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultSessionStorage is the default bbolt database implementation of the
|
||||||
|
// [SessionStorage] interface.
|
||||||
|
type DefaultSessionStorage struct {
|
||||||
|
// db is an instance of the bbolt database where web user sessions are
|
||||||
|
// stored by [SessionToken] in the [bucketNameSessions] bucket.
|
||||||
|
db *bbolt.DB
|
||||||
|
|
||||||
|
// logger is used for logging the operation of the session storage.
|
||||||
|
logger *slog.Logger
|
||||||
|
|
||||||
|
// mu protects sessions.
|
||||||
|
mu *sync.Mutex
|
||||||
|
|
||||||
|
// clock is used to get the current time.
|
||||||
|
clock timeutil.Clock
|
||||||
|
|
||||||
|
// userDB contains the web user information such as ID, login, and password.
|
||||||
|
userDB DB
|
||||||
|
|
||||||
|
// sessions maps a session token to a web user session.
|
||||||
|
sessions map[SessionToken]*Session
|
||||||
|
|
||||||
|
// sessionTTL is the default Time-To-Live value for web user sessions.
|
||||||
|
sessionTTL time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewDefaultSessionStorage returns the new properly initialized
|
||||||
|
// *DefaultSessionStorage.
|
||||||
|
func NewDefaultSessionStorage(
|
||||||
|
ctx context.Context,
|
||||||
|
conf *DefaultSessionStorageConfig,
|
||||||
|
) (ds *DefaultSessionStorage, err error) {
|
||||||
|
ds = &DefaultSessionStorage{
|
||||||
|
clock: conf.Clock,
|
||||||
|
userDB: conf.UserDB,
|
||||||
|
logger: conf.Logger,
|
||||||
|
mu: &sync.Mutex{},
|
||||||
|
sessions: map[SessionToken]*Session{},
|
||||||
|
sessionTTL: conf.SessionTTL,
|
||||||
|
}
|
||||||
|
|
||||||
|
dbFilename := conf.DBPath
|
||||||
|
// TODO(s.chzhen): Pass logger with options.
|
||||||
|
ds.db, err = bbolt.Open(dbFilename, aghos.DefaultPermFile, nil)
|
||||||
|
if err != nil {
|
||||||
|
ds.logger.ErrorContext(ctx, "opening db %q: %w", dbFilename, err)
|
||||||
|
if errors.Is(err, berrors.ErrInvalid) {
|
||||||
|
const s = "AdGuard Home cannot be initialized due to an incompatible file system.\n" +
|
||||||
|
"Please read the explanation here: https://adguard-dns.io/kb/adguard-home/getting-started/#limitations"
|
||||||
|
slogutil.PrintLines(ctx, ds.logger, slog.LevelError, "", s)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = ds.loadSessions(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("loading sessions: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return ds, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// loadSessions loads web user sessions from the bbolt database.
|
||||||
|
func (ds *DefaultSessionStorage) loadSessions(ctx context.Context) (err error) {
|
||||||
|
tx, err := ds.db.Begin(true)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("starting transaction: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
needRollback := true
|
||||||
|
defer func() {
|
||||||
|
if needRollback {
|
||||||
|
err = errors.WithDeferred(err, tx.Rollback())
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
bkt := tx.Bucket([]byte(bboltBucketSessions))
|
||||||
|
if bkt == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
removed, err := ds.processSessions(ctx, bkt)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("processing sessions: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if removed == 0 {
|
||||||
|
ds.logger.DebugContext(ctx, "loading sessions from db", "stored", len(ds.sessions))
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
needRollback = false
|
||||||
|
err = tx.Commit()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("committing transaction: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ds.logger.DebugContext(
|
||||||
|
ctx,
|
||||||
|
"loading sessions from db",
|
||||||
|
"stored", len(ds.sessions),
|
||||||
|
"removed", removed,
|
||||||
|
)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// processSessions iterates over the sessions bucket and loads or removes
|
||||||
|
// sessions as needed.
|
||||||
|
func (ds *DefaultSessionStorage) processSessions(
|
||||||
|
ctx context.Context,
|
||||||
|
bkt *bbolt.Bucket,
|
||||||
|
) (removed int, err error) {
|
||||||
|
invalidSessions := [][]byte{}
|
||||||
|
|
||||||
|
err = bkt.ForEach(ds.bboltSessionHandler(ctx, &invalidSessions))
|
||||||
|
if err != nil {
|
||||||
|
return 0, fmt.Errorf("iterating over sessions: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var errs []error
|
||||||
|
for _, s := range invalidSessions {
|
||||||
|
if err = bkt.Delete(s); err != nil {
|
||||||
|
errs = append(errs, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = errors.Join(errs...); err != nil {
|
||||||
|
return 0, fmt.Errorf("deleting sessions: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return len(invalidSessions), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// bboltSessionHandler returns a function for [bbolt.Bucket.ForEach] that
|
||||||
|
// iterates over stored sessions, deserializes them, and logs any errors
|
||||||
|
// encountered. The returned error is always nil, as these errors are
|
||||||
|
// considered non-critical to stop the iteration process.
|
||||||
|
func (ds *DefaultSessionStorage) bboltSessionHandler(
|
||||||
|
ctx context.Context,
|
||||||
|
invalidSessions *[][]byte,
|
||||||
|
) (fn func(k, v []byte) (err error)) {
|
||||||
|
now := ds.clock.Now()
|
||||||
|
|
||||||
|
return func(k, v []byte) (err error) {
|
||||||
|
s, err := bboltDecode(v)
|
||||||
|
if err != nil {
|
||||||
|
*invalidSessions = append(*invalidSessions, k)
|
||||||
|
ds.logger.DebugContext(ctx, "deserializing session", slogutil.KeyError, err)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if now.After(s.Expire) {
|
||||||
|
*invalidSessions = append(*invalidSessions, k)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
u, err := ds.userDB.ByLogin(ctx, s.UserLogin)
|
||||||
|
if err != nil {
|
||||||
|
// Should not happen, as it currently always returns nil for error.
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if u == nil {
|
||||||
|
*invalidSessions = append(*invalidSessions, k)
|
||||||
|
ds.logger.DebugContext(ctx, "no saved user by name", "name", s.UserLogin)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
t := SessionToken(k)
|
||||||
|
s.Token = t
|
||||||
|
s.UserID = u.ID
|
||||||
|
ds.sessions[t] = s
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// bboltBucketSessions is the name of the bucket storing web user sessions in
|
||||||
|
// the bbolt database.
|
||||||
|
const bboltBucketSessions = "sessions-2"
|
||||||
|
|
||||||
|
const (
|
||||||
|
// bboltSessionExpireLen is the length of the expire field in the binary
|
||||||
|
// entry stored in bbolt.
|
||||||
|
bboltSessionExpireLen = 4
|
||||||
|
|
||||||
|
// bboltSessionNameLen is the length of the name field in the binary entry
|
||||||
|
// stored in bbolt.
|
||||||
|
bboltSessionNameLen = 2
|
||||||
|
)
|
||||||
|
|
||||||
|
// bboltDecode deserializes decodes a binary data into a session.
|
||||||
|
func bboltDecode(data []byte) (s *Session, err error) {
|
||||||
|
if len(data) < bboltSessionExpireLen+bboltSessionNameLen {
|
||||||
|
return nil, fmt.Errorf("length of the data is less than expected: got %d", len(data))
|
||||||
|
}
|
||||||
|
|
||||||
|
expireData := data[:bboltSessionExpireLen]
|
||||||
|
nameLenData := data[bboltSessionExpireLen : bboltSessionExpireLen+bboltSessionNameLen]
|
||||||
|
nameData := data[bboltSessionExpireLen+bboltSessionNameLen:]
|
||||||
|
|
||||||
|
nameLen := binary.BigEndian.Uint16(nameLenData)
|
||||||
|
if len(nameData) != int(nameLen) {
|
||||||
|
return nil, fmt.Errorf("login: expected length %d, got %d", nameLen, len(nameData))
|
||||||
|
}
|
||||||
|
|
||||||
|
expire := binary.BigEndian.Uint32(expireData)
|
||||||
|
|
||||||
|
return &Session{
|
||||||
|
Expire: time.Unix(int64(expire), 0),
|
||||||
|
UserLogin: Login(nameData),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// bboltEncode serializes a session properties into a binary data.
|
||||||
|
func bboltEncode(s *Session) (data []byte) {
|
||||||
|
data = make([]byte, bboltSessionExpireLen+bboltSessionNameLen+len(s.UserLogin))
|
||||||
|
|
||||||
|
expireData := data[:bboltSessionExpireLen]
|
||||||
|
nameLenData := data[bboltSessionExpireLen : bboltSessionExpireLen+bboltSessionNameLen]
|
||||||
|
nameData := data[bboltSessionExpireLen+bboltSessionNameLen:]
|
||||||
|
|
||||||
|
expire := uint32(s.Expire.Unix())
|
||||||
|
binary.BigEndian.PutUint32(expireData, expire)
|
||||||
|
binary.BigEndian.PutUint16(nameLenData, uint16(len(s.UserLogin)))
|
||||||
|
copy(nameData, []byte(s.UserLogin))
|
||||||
|
|
||||||
|
return data
|
||||||
|
}
|
||||||
|
|
||||||
|
// type check
|
||||||
|
var _ SessionStorage = (*DefaultSessionStorage)(nil)
|
||||||
|
|
||||||
|
// New implements the [SessionStorage] interface for *DefaultSessionStorage.
|
||||||
|
func (ds *DefaultSessionStorage) New(ctx context.Context, u *User) (s *Session, err error) {
|
||||||
|
s = &Session{
|
||||||
|
Token: NewSessionToken(),
|
||||||
|
UserID: u.ID,
|
||||||
|
UserLogin: u.Login,
|
||||||
|
Expire: ds.clock.Now().Add(ds.sessionTTL),
|
||||||
|
}
|
||||||
|
|
||||||
|
err = ds.store(s)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("storing session: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ds.mu.Lock()
|
||||||
|
defer ds.mu.Unlock()
|
||||||
|
|
||||||
|
ds.sessions[s.Token] = s
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// store saves a web user session in the bbolt database.
|
||||||
|
func (ds *DefaultSessionStorage) store(s *Session) (err error) {
|
||||||
|
tx, err := ds.db.Begin(true)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("starting transaction: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
needRollback := true
|
||||||
|
defer func() {
|
||||||
|
if needRollback {
|
||||||
|
err = errors.WithDeferred(err, tx.Rollback())
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
bkt, err := tx.CreateBucketIfNotExists([]byte(bboltBucketSessions))
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("creating bucket: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = bkt.Put(s.Token[:], bboltEncode(s))
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("putting data: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
needRollback = false
|
||||||
|
err = tx.Commit()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("committing transaction: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FindByToken implements the [SessionStorage] interface for *DefaultSessionStorage.
|
||||||
|
func (ds *DefaultSessionStorage) FindByToken(ctx context.Context, t SessionToken) (s *Session, err error) {
|
||||||
|
ds.mu.Lock()
|
||||||
|
defer ds.mu.Unlock()
|
||||||
|
|
||||||
|
s, ok := ds.sessions[t]
|
||||||
|
if !ok {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
now := ds.clock.Now()
|
||||||
|
if now.After(s.Expire) {
|
||||||
|
err = ds.deleteByToken(ctx, t)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("expired session: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeleteByToken implements the [SessionStorage] interface for
|
||||||
|
// *DefaultSessionStorage.
|
||||||
|
func (ds *DefaultSessionStorage) DeleteByToken(ctx context.Context, t SessionToken) (err error) {
|
||||||
|
ds.mu.Lock()
|
||||||
|
defer ds.mu.Unlock()
|
||||||
|
|
||||||
|
// Don't wrap the error because it's informative enough as is.
|
||||||
|
return ds.deleteByToken(ctx, t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// deleteByToken removes stored session by token. ds.mu is expected to be
|
||||||
|
// locked.
|
||||||
|
func (ds *DefaultSessionStorage) deleteByToken(ctx context.Context, t SessionToken) (err error) {
|
||||||
|
err = ds.remove(ctx, t)
|
||||||
|
if err != nil {
|
||||||
|
ds.logger.ErrorContext(ctx, "deleting session", slogutil.KeyError, err)
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
delete(ds.sessions, t)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove deletes a web user session from the bbolt database.
|
||||||
|
func (ds *DefaultSessionStorage) remove(ctx context.Context, t SessionToken) (err error) {
|
||||||
|
tx, err := ds.db.Begin(true)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("starting transaction: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
needRollback := true
|
||||||
|
defer func() {
|
||||||
|
if needRollback {
|
||||||
|
err = errors.WithDeferred(err, tx.Rollback())
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
bkt := tx.Bucket([]byte(bboltBucketSessions))
|
||||||
|
if bkt == nil {
|
||||||
|
return errors.Error("no bucket")
|
||||||
|
}
|
||||||
|
|
||||||
|
err = bkt.Delete(t[:])
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("removing data: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
needRollback = false
|
||||||
|
err = tx.Commit()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("committing transaction: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ds.logger.DebugContext(ctx, "removed session from db")
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close implements the [SessionStorage] interface for *DefaultSessionStorage.
|
||||||
|
func (ds *DefaultSessionStorage) Close() (err error) {
|
||||||
|
err = ds.db.Close()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("closing db: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
162
internal/aghuser/sessionstorage_test.go
Normal file
162
internal/aghuser/sessionstorage_test.go
Normal file
@@ -0,0 +1,162 @@
|
|||||||
|
package aghuser_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/AdguardTeam/AdGuardHome/internal/aghuser"
|
||||||
|
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
||||||
|
"github.com/AdguardTeam/golibs/testutil"
|
||||||
|
"github.com/AdguardTeam/golibs/testutil/faketime"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
// addSession is a helper function that saves and returns a session for a newly
|
||||||
|
// generated [aghuser.User] by login.
|
||||||
|
func addSession(
|
||||||
|
tb testing.TB,
|
||||||
|
ctx context.Context,
|
||||||
|
ds aghuser.SessionStorage,
|
||||||
|
login aghuser.Login,
|
||||||
|
) (s *aghuser.Session) {
|
||||||
|
tb.Helper()
|
||||||
|
|
||||||
|
s, err := ds.New(ctx, &aghuser.User{
|
||||||
|
ID: aghuser.MustNewUserID(),
|
||||||
|
Login: login,
|
||||||
|
})
|
||||||
|
require.NoError(tb, err)
|
||||||
|
require.NotNil(tb, s)
|
||||||
|
|
||||||
|
var got *aghuser.Session
|
||||||
|
got, err = ds.FindByToken(ctx, s.Token)
|
||||||
|
require.NoError(tb, err)
|
||||||
|
require.NotNil(tb, got)
|
||||||
|
|
||||||
|
assert.Equal(tb, login, got.UserLogin)
|
||||||
|
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDefaultSessionStorage(t *testing.T) {
|
||||||
|
const (
|
||||||
|
userLoginFirst aghuser.Login = "user_one"
|
||||||
|
userLoginSecond aghuser.Login = "user_two"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
ctx = testutil.ContextWithTimeout(t, testTimeout)
|
||||||
|
logger = slogutil.NewDiscardLogger()
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
sessionTTL = time.Minute
|
||||||
|
timeStep = time.Second
|
||||||
|
)
|
||||||
|
|
||||||
|
// Set up a mock clock to test expired sessions. Each call to [clock.Now]
|
||||||
|
// will return the [date] incremented by [timeStep].
|
||||||
|
date := time.Now()
|
||||||
|
clock := &faketime.Clock{
|
||||||
|
OnNow: func() (now time.Time) {
|
||||||
|
date = date.Add(timeStep)
|
||||||
|
|
||||||
|
return date
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
dbFile, err := os.CreateTemp(t.TempDir(), "sessions.db")
|
||||||
|
require.NoError(t, err)
|
||||||
|
testutil.CleanupAndRequireSuccess(t, dbFile.Close)
|
||||||
|
|
||||||
|
userDB := aghuser.NewDefaultDB()
|
||||||
|
|
||||||
|
err = userDB.Create(ctx, &aghuser.User{
|
||||||
|
Login: userLoginFirst,
|
||||||
|
ID: aghuser.MustNewUserID(),
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
err = userDB.Create(ctx, &aghuser.User{
|
||||||
|
Login: userLoginSecond,
|
||||||
|
ID: aghuser.MustNewUserID(),
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
var (
|
||||||
|
ds *aghuser.DefaultSessionStorage
|
||||||
|
|
||||||
|
sessionFirst *aghuser.Session
|
||||||
|
sessionSecond *aghuser.Session
|
||||||
|
)
|
||||||
|
|
||||||
|
require.True(t, t.Run("prepare_session_storage", func(t *testing.T) {
|
||||||
|
ds, err = aghuser.NewDefaultSessionStorage(ctx, &aghuser.DefaultSessionStorageConfig{
|
||||||
|
Clock: clock,
|
||||||
|
UserDB: userDB,
|
||||||
|
Logger: logger,
|
||||||
|
DBPath: dbFile.Name(),
|
||||||
|
SessionTTL: sessionTTL,
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
sessionFirst = addSession(t, ctx, ds, userLoginFirst)
|
||||||
|
|
||||||
|
// Advance time to ensure the first session expires before creating the
|
||||||
|
// second session.
|
||||||
|
date = date.Add(time.Hour)
|
||||||
|
|
||||||
|
sessionSecond = addSession(t, ctx, ds, userLoginSecond)
|
||||||
|
|
||||||
|
err = ds.Close()
|
||||||
|
require.NoError(t, err)
|
||||||
|
}))
|
||||||
|
|
||||||
|
require.True(t, t.Run("load_sessions", func(t *testing.T) {
|
||||||
|
ds, err = aghuser.NewDefaultSessionStorage(ctx, &aghuser.DefaultSessionStorageConfig{
|
||||||
|
Clock: clock,
|
||||||
|
UserDB: userDB,
|
||||||
|
Logger: logger,
|
||||||
|
DBPath: dbFile.Name(),
|
||||||
|
SessionTTL: sessionTTL,
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
var got *aghuser.Session
|
||||||
|
got, err = ds.FindByToken(ctx, sessionFirst.Token)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
assert.Nil(t, got)
|
||||||
|
|
||||||
|
got, err = ds.FindByToken(ctx, sessionSecond.Token)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.NotNil(t, got)
|
||||||
|
|
||||||
|
assert.Equal(t, userLoginSecond, got.UserLogin)
|
||||||
|
|
||||||
|
err = ds.DeleteByToken(ctx, sessionSecond.Token)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
got, err = ds.FindByToken(ctx, sessionSecond.Token)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
assert.Nil(t, got)
|
||||||
|
}))
|
||||||
|
|
||||||
|
require.True(t, t.Run("expired_session", func(t *testing.T) {
|
||||||
|
testutil.CleanupAndRequireSuccess(t, ds.Close)
|
||||||
|
|
||||||
|
sessionFirst = addSession(t, ctx, ds, userLoginFirst)
|
||||||
|
|
||||||
|
date = date.Add(time.Hour)
|
||||||
|
|
||||||
|
var got *aghuser.Session
|
||||||
|
got, err = ds.FindByToken(ctx, sessionFirst.Token)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
assert.Nil(t, got)
|
||||||
|
}))
|
||||||
|
}
|
||||||
@@ -32,13 +32,13 @@ func MustNewUserID() (uid UserID) {
|
|||||||
|
|
||||||
// User represents a web user.
|
// User represents a web user.
|
||||||
type User struct {
|
type User struct {
|
||||||
// ID is the unique identifier for the web user. It must not be empty.
|
// Password stores the password information for the web user. It must not
|
||||||
ID UserID
|
// be nil.
|
||||||
|
Password Password
|
||||||
|
|
||||||
// Login is the login name of the web user. It must not be empty.
|
// Login is the login name of the web user. It must not be empty.
|
||||||
Login Login
|
Login Login
|
||||||
|
|
||||||
// Password stores the password information for the web user. It must not
|
// ID is the unique identifier for the web user. It must not be empty.
|
||||||
// be nil.
|
ID UserID
|
||||||
Password Password
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,8 +11,34 @@ import (
|
|||||||
"slices"
|
"slices"
|
||||||
|
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/whois"
|
"github.com/AdguardTeam/AdGuardHome/internal/whois"
|
||||||
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
|
"github.com/AdguardTeam/golibs/netutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// ClientID is a unique identifier for a persistent client used in
|
||||||
|
// DNS-over-HTTPS, DNS-over-TLS, and DNS-over-QUIC queries.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Use everywhere.
|
||||||
|
type ClientID string
|
||||||
|
|
||||||
|
// ValidateClientID returns an error if id is not a valid ClientID.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Consider implementing [validate.Interface] for ClientID.
|
||||||
|
func ValidateClientID(id string) (err error) {
|
||||||
|
err = netutil.ValidateHostnameLabel(id)
|
||||||
|
if err != nil {
|
||||||
|
// Replace the domain name label wrapper with our own.
|
||||||
|
return fmt.Errorf("invalid clientid %q: %w", id, errors.Unwrap(err))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// isValidClientID returns false if id is not a valid ClientID.
|
||||||
|
func isValidClientID(id string) (ok bool) {
|
||||||
|
return netutil.IsValidHostnameLabel(id)
|
||||||
|
}
|
||||||
|
|
||||||
// Source represents the source from which the information about the client has
|
// Source represents the source from which the information about the client has
|
||||||
// been obtained.
|
// been obtained.
|
||||||
type Source uint8
|
type Source uint8
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ type index struct {
|
|||||||
nameToUID map[string]UID
|
nameToUID map[string]UID
|
||||||
|
|
||||||
// clientIDToUID maps ClientID to UID.
|
// clientIDToUID maps ClientID to UID.
|
||||||
clientIDToUID map[string]UID
|
clientIDToUID map[ClientID]UID
|
||||||
|
|
||||||
// ipToUID maps IP address to UID.
|
// ipToUID maps IP address to UID.
|
||||||
ipToUID map[netip.Addr]UID
|
ipToUID map[netip.Addr]UID
|
||||||
@@ -54,7 +54,7 @@ type index struct {
|
|||||||
func newIndex() (ci *index) {
|
func newIndex() (ci *index) {
|
||||||
return &index{
|
return &index{
|
||||||
nameToUID: map[string]UID{},
|
nameToUID: map[string]UID{},
|
||||||
clientIDToUID: map[string]UID{},
|
clientIDToUID: map[ClientID]UID{},
|
||||||
ipToUID: map[netip.Addr]UID{},
|
ipToUID: map[netip.Addr]UID{},
|
||||||
subnetToUID: aghalg.NewSortedMap[netip.Prefix, UID](subnetCompare),
|
subnetToUID: aghalg.NewSortedMap[netip.Prefix, UID](subnetCompare),
|
||||||
macToUID: map[macKey]UID{},
|
macToUID: map[macKey]UID{},
|
||||||
@@ -207,7 +207,7 @@ func (ci *index) clashesMAC(c *Persistent) (p *Persistent, mac net.HardwareAddr)
|
|||||||
// find finds persistent client by string representation of the ClientID, IP
|
// find finds persistent client by string representation of the ClientID, IP
|
||||||
// address, or MAC.
|
// address, or MAC.
|
||||||
func (ci *index) find(id string) (c *Persistent, ok bool) {
|
func (ci *index) find(id string) (c *Persistent, ok bool) {
|
||||||
c, ok = ci.findByClientID(id)
|
c, ok = ci.findByClientID(ClientID(id))
|
||||||
if ok {
|
if ok {
|
||||||
return c, true
|
return c, true
|
||||||
}
|
}
|
||||||
@@ -230,7 +230,7 @@ func (ci *index) find(id string) (c *Persistent, ok bool) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// findByClientID finds persistent client by ClientID.
|
// findByClientID finds persistent client by ClientID.
|
||||||
func (ci *index) findByClientID(clientID string) (c *Persistent, ok bool) {
|
func (ci *index) findByClientID(clientID ClientID) (c *Persistent, ok bool) {
|
||||||
uid, ok := ci.clientIDToUID[clientID]
|
uid, ok := ci.clientIDToUID[clientID]
|
||||||
if ok {
|
if ok {
|
||||||
return ci.uidToClient[uid], true
|
return ci.uidToClient[uid], true
|
||||||
@@ -275,6 +275,26 @@ func (ci *index) findByIP(ip netip.Addr) (c *Persistent, found bool) {
|
|||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// findByCIDR searches for a persistent client with the provided subnet as an
|
||||||
|
// identifier. Note that this function looks for an exact match of subnets,
|
||||||
|
// rather than checking if one subnet contains another.
|
||||||
|
func (ci *index) findByCIDR(subnet netip.Prefix) (c *Persistent, ok bool) {
|
||||||
|
var uid UID
|
||||||
|
for pref, id := range ci.subnetToUID.Range {
|
||||||
|
if subnet == pref {
|
||||||
|
uid, ok = id, true
|
||||||
|
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ok {
|
||||||
|
return ci.uidToClient[uid], true
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
|
||||||
// findByMAC finds persistent client by MAC.
|
// findByMAC finds persistent client by MAC.
|
||||||
func (ci *index) findByMAC(mac net.HardwareAddr) (c *Persistent, found bool) {
|
func (ci *index) findByMAC(mac net.HardwareAddr) (c *Persistent, found bool) {
|
||||||
k := macToKey(mac)
|
k := macToKey(mac)
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import (
|
|||||||
"net/netip"
|
"net/netip"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
@@ -58,12 +59,12 @@ func TestClientIndex_Find(t *testing.T) {
|
|||||||
|
|
||||||
clientWithMAC = &Persistent{
|
clientWithMAC = &Persistent{
|
||||||
Name: "client_with_mac",
|
Name: "client_with_mac",
|
||||||
MACs: []net.HardwareAddr{mustParseMAC(cliMAC)},
|
MACs: []net.HardwareAddr{errors.Must(net.ParseMAC(cliMAC))},
|
||||||
}
|
}
|
||||||
|
|
||||||
clientWithID = &Persistent{
|
clientWithID = &Persistent{
|
||||||
Name: "client_with_id",
|
Name: "client_with_id",
|
||||||
ClientIDs: []string{cliID},
|
ClientIDs: []ClientID{cliID},
|
||||||
}
|
}
|
||||||
|
|
||||||
clientLinkLocal = &Persistent{
|
clientLinkLocal = &Persistent{
|
||||||
@@ -141,10 +142,10 @@ func TestClientIndex_Clashes(t *testing.T) {
|
|||||||
Subnets: []netip.Prefix{netip.MustParsePrefix(cliSubnet)},
|
Subnets: []netip.Prefix{netip.MustParsePrefix(cliSubnet)},
|
||||||
}, {
|
}, {
|
||||||
Name: "client_with_mac",
|
Name: "client_with_mac",
|
||||||
MACs: []net.HardwareAddr{mustParseMAC(cliMAC)},
|
MACs: []net.HardwareAddr{errors.Must(net.ParseMAC(cliMAC))},
|
||||||
}, {
|
}, {
|
||||||
Name: "client_with_id",
|
Name: "client_with_id",
|
||||||
ClientIDs: []string{cliID},
|
ClientIDs: []ClientID{cliID},
|
||||||
}}
|
}}
|
||||||
|
|
||||||
ci := newIDIndex(clients)
|
ci := newIDIndex(clients)
|
||||||
@@ -181,17 +182,6 @@ func TestClientIndex_Clashes(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// mustParseMAC is wrapper around [net.ParseMAC] that panics if there is an
|
|
||||||
// error.
|
|
||||||
func mustParseMAC(s string) (mac net.HardwareAddr) {
|
|
||||||
mac, err := net.ParseMAC(s)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return mac
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestMACToKey(t *testing.T) {
|
func TestMACToKey(t *testing.T) {
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
want any
|
want any
|
||||||
@@ -200,44 +190,44 @@ func TestMACToKey(t *testing.T) {
|
|||||||
}{{
|
}{{
|
||||||
name: "column6",
|
name: "column6",
|
||||||
in: "00:00:5e:00:53:01",
|
in: "00:00:5e:00:53:01",
|
||||||
want: [6]byte(mustParseMAC("00:00:5e:00:53:01")),
|
want: [6]byte(errors.Must(net.ParseMAC("00:00:5e:00:53:01"))),
|
||||||
}, {
|
}, {
|
||||||
name: "column8",
|
name: "column8",
|
||||||
in: "02:00:5e:10:00:00:00:01",
|
in: "02:00:5e:10:00:00:00:01",
|
||||||
want: [8]byte(mustParseMAC("02:00:5e:10:00:00:00:01")),
|
want: [8]byte(errors.Must(net.ParseMAC("02:00:5e:10:00:00:00:01"))),
|
||||||
}, {
|
}, {
|
||||||
name: "column20",
|
name: "column20",
|
||||||
in: "00:00:00:00:fe:80:00:00:00:00:00:00:02:00:5e:10:00:00:00:01",
|
in: "00:00:00:00:fe:80:00:00:00:00:00:00:02:00:5e:10:00:00:00:01",
|
||||||
want: [20]byte(mustParseMAC("00:00:00:00:fe:80:00:00:00:00:00:00:02:00:5e:10:00:00:00:01")),
|
want: [20]byte(errors.Must(net.ParseMAC("00:00:00:00:fe:80:00:00:00:00:00:00:02:00:5e:10:00:00:00:01"))),
|
||||||
}, {
|
}, {
|
||||||
name: "hyphen6",
|
name: "hyphen6",
|
||||||
in: "00-00-5e-00-53-01",
|
in: "00-00-5e-00-53-01",
|
||||||
want: [6]byte(mustParseMAC("00-00-5e-00-53-01")),
|
want: [6]byte(errors.Must(net.ParseMAC("00-00-5e-00-53-01"))),
|
||||||
}, {
|
}, {
|
||||||
name: "hyphen8",
|
name: "hyphen8",
|
||||||
in: "02-00-5e-10-00-00-00-01",
|
in: "02-00-5e-10-00-00-00-01",
|
||||||
want: [8]byte(mustParseMAC("02-00-5e-10-00-00-00-01")),
|
want: [8]byte(errors.Must(net.ParseMAC("02-00-5e-10-00-00-00-01"))),
|
||||||
}, {
|
}, {
|
||||||
name: "hyphen20",
|
name: "hyphen20",
|
||||||
in: "00-00-00-00-fe-80-00-00-00-00-00-00-02-00-5e-10-00-00-00-01",
|
in: "00-00-00-00-fe-80-00-00-00-00-00-00-02-00-5e-10-00-00-00-01",
|
||||||
want: [20]byte(mustParseMAC("00-00-00-00-fe-80-00-00-00-00-00-00-02-00-5e-10-00-00-00-01")),
|
want: [20]byte(errors.Must(net.ParseMAC("00-00-00-00-fe-80-00-00-00-00-00-00-02-00-5e-10-00-00-00-01"))),
|
||||||
}, {
|
}, {
|
||||||
name: "dot6",
|
name: "dot6",
|
||||||
in: "0000.5e00.5301",
|
in: "0000.5e00.5301",
|
||||||
want: [6]byte(mustParseMAC("0000.5e00.5301")),
|
want: [6]byte(errors.Must(net.ParseMAC("0000.5e00.5301"))),
|
||||||
}, {
|
}, {
|
||||||
name: "dot8",
|
name: "dot8",
|
||||||
in: "0200.5e10.0000.0001",
|
in: "0200.5e10.0000.0001",
|
||||||
want: [8]byte(mustParseMAC("0200.5e10.0000.0001")),
|
want: [8]byte(errors.Must(net.ParseMAC("0200.5e10.0000.0001"))),
|
||||||
}, {
|
}, {
|
||||||
name: "dot20",
|
name: "dot20",
|
||||||
in: "0000.0000.fe80.0000.0000.0000.0200.5e10.0000.0001",
|
in: "0000.0000.fe80.0000.0000.0000.0200.5e10.0000.0001",
|
||||||
want: [20]byte(mustParseMAC("0000.0000.fe80.0000.0000.0000.0200.5e10.0000.0001")),
|
want: [20]byte(errors.Must(net.ParseMAC("0000.0000.fe80.0000.0000.0000.0200.5e10.0000.0001"))),
|
||||||
}}
|
}}
|
||||||
|
|
||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
mac := mustParseMAC(tc.in)
|
mac := errors.Must(net.ParseMAC(tc.in))
|
||||||
|
|
||||||
key := macToKey(mac)
|
key := macToKey(mac)
|
||||||
assert.Equal(t, tc.want, key)
|
assert.Equal(t, tc.want, key)
|
||||||
@@ -302,19 +292,19 @@ func TestIndex_FindByIPWithoutZone(t *testing.T) {
|
|||||||
func TestClientIndex_RangeByName(t *testing.T) {
|
func TestClientIndex_RangeByName(t *testing.T) {
|
||||||
sortedClients := []*Persistent{{
|
sortedClients := []*Persistent{{
|
||||||
Name: "clientA",
|
Name: "clientA",
|
||||||
ClientIDs: []string{"A"},
|
ClientIDs: []ClientID{"A"},
|
||||||
}, {
|
}, {
|
||||||
Name: "clientB",
|
Name: "clientB",
|
||||||
ClientIDs: []string{"B"},
|
ClientIDs: []ClientID{"B"},
|
||||||
}, {
|
}, {
|
||||||
Name: "clientC",
|
Name: "clientC",
|
||||||
ClientIDs: []string{"C"},
|
ClientIDs: []ClientID{"C"},
|
||||||
}, {
|
}, {
|
||||||
Name: "clientD",
|
Name: "clientD",
|
||||||
ClientIDs: []string{"D"},
|
ClientIDs: []ClientID{"D"},
|
||||||
}, {
|
}, {
|
||||||
Name: "clientE",
|
Name: "clientE",
|
||||||
ClientIDs: []string{"E"},
|
ClientIDs: []ClientID{"E"},
|
||||||
}}
|
}}
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
@@ -349,3 +339,115 @@ func TestClientIndex_RangeByName(t *testing.T) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestIndex_FindByName(t *testing.T) {
|
||||||
|
const (
|
||||||
|
clientExistingName = "client_existing"
|
||||||
|
clientAnotherExistingName = "client_another_existing"
|
||||||
|
nonExistingClientName = "client_non_existing"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
clientExisting = &Persistent{
|
||||||
|
Name: clientExistingName,
|
||||||
|
IPs: []netip.Addr{netip.MustParseAddr("192.0.2.1")},
|
||||||
|
}
|
||||||
|
|
||||||
|
clientAnotherExisting = &Persistent{
|
||||||
|
Name: clientAnotherExistingName,
|
||||||
|
IPs: []netip.Addr{netip.MustParseAddr("192.0.2.2")},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
clients := []*Persistent{
|
||||||
|
clientExisting,
|
||||||
|
clientAnotherExisting,
|
||||||
|
}
|
||||||
|
ci := newIDIndex(clients)
|
||||||
|
|
||||||
|
testCases := []struct {
|
||||||
|
want *Persistent
|
||||||
|
found assert.BoolAssertionFunc
|
||||||
|
name string
|
||||||
|
clientName string
|
||||||
|
}{{
|
||||||
|
want: clientExisting,
|
||||||
|
found: assert.True,
|
||||||
|
name: "existing",
|
||||||
|
clientName: clientExistingName,
|
||||||
|
}, {
|
||||||
|
want: clientAnotherExisting,
|
||||||
|
found: assert.True,
|
||||||
|
name: "another_existing",
|
||||||
|
clientName: clientAnotherExistingName,
|
||||||
|
}, {
|
||||||
|
want: nil,
|
||||||
|
found: assert.False,
|
||||||
|
name: "non_existing",
|
||||||
|
clientName: nonExistingClientName,
|
||||||
|
}}
|
||||||
|
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
c, ok := ci.findByName(tc.clientName)
|
||||||
|
assert.Equal(t, tc.want, c)
|
||||||
|
tc.found(t, ok)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestIndex_FindByMAC(t *testing.T) {
|
||||||
|
var (
|
||||||
|
cliMAC = errors.Must(net.ParseMAC("11:11:11:11:11:11"))
|
||||||
|
cliAnotherMAC = errors.Must(net.ParseMAC("22:22:22:22:22:22"))
|
||||||
|
nonExistingClientMAC = errors.Must(net.ParseMAC("33:33:33:33:33:33"))
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
clientExisting = &Persistent{
|
||||||
|
Name: "client",
|
||||||
|
MACs: []net.HardwareAddr{cliMAC},
|
||||||
|
}
|
||||||
|
|
||||||
|
clientAnotherExisting = &Persistent{
|
||||||
|
Name: "another_client",
|
||||||
|
MACs: []net.HardwareAddr{cliAnotherMAC},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
clients := []*Persistent{
|
||||||
|
clientExisting,
|
||||||
|
clientAnotherExisting,
|
||||||
|
}
|
||||||
|
ci := newIDIndex(clients)
|
||||||
|
|
||||||
|
testCases := []struct {
|
||||||
|
want *Persistent
|
||||||
|
found assert.BoolAssertionFunc
|
||||||
|
name string
|
||||||
|
clientMAC net.HardwareAddr
|
||||||
|
}{{
|
||||||
|
want: clientExisting,
|
||||||
|
found: assert.True,
|
||||||
|
name: "existing",
|
||||||
|
clientMAC: cliMAC,
|
||||||
|
}, {
|
||||||
|
want: clientAnotherExisting,
|
||||||
|
found: assert.True,
|
||||||
|
name: "another_existing",
|
||||||
|
clientMAC: cliAnotherMAC,
|
||||||
|
}, {
|
||||||
|
want: nil,
|
||||||
|
found: assert.False,
|
||||||
|
name: "non_existing",
|
||||||
|
clientMAC: nonExistingClientMAC,
|
||||||
|
}}
|
||||||
|
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
c, ok := ci.findByMAC(tc.clientMAC)
|
||||||
|
assert.Equal(t, tc.want, c)
|
||||||
|
tc.found(t, ok)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -15,7 +15,6 @@ import (
|
|||||||
"github.com/AdguardTeam/dnsproxy/upstream"
|
"github.com/AdguardTeam/dnsproxy/upstream"
|
||||||
"github.com/AdguardTeam/golibs/errors"
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
||||||
"github.com/AdguardTeam/golibs/netutil"
|
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -71,7 +70,9 @@ type Persistent struct {
|
|||||||
// Tags is a list of client tags that categorize the client.
|
// Tags is a list of client tags that categorize the client.
|
||||||
Tags []string
|
Tags []string
|
||||||
|
|
||||||
// Upstreams is a list of custom upstream DNS servers for the client.
|
// Upstreams is a list of custom upstream DNS servers for the client. If
|
||||||
|
// it's empty, the custom upstream cache is disabled, regardless of the
|
||||||
|
// value of UpstreamsCacheEnabled.
|
||||||
Upstreams []string
|
Upstreams []string
|
||||||
|
|
||||||
// IPs is a list of IP addresses that identify the client. The client must
|
// IPs is a list of IP addresses that identify the client. The client must
|
||||||
@@ -90,15 +91,16 @@ type Persistent struct {
|
|||||||
|
|
||||||
// ClientIDs identifying the client. The client must have at least one ID
|
// ClientIDs identifying the client. The client must have at least one ID
|
||||||
// (IP, subnet, MAC, or ClientID).
|
// (IP, subnet, MAC, or ClientID).
|
||||||
ClientIDs []string
|
ClientIDs []ClientID
|
||||||
|
|
||||||
// UID is the unique identifier of the persistent client.
|
// UID is the unique identifier of the persistent client.
|
||||||
UID UID
|
UID UID
|
||||||
|
|
||||||
// UpstreamsCacheSize is the cache size for custom upstreams.
|
// UpstreamsCacheSize defines the size of the custom upstream cache.
|
||||||
UpstreamsCacheSize uint32
|
UpstreamsCacheSize uint32
|
||||||
|
|
||||||
// UpstreamsCacheEnabled specifies whether custom upstreams are used.
|
// UpstreamsCacheEnabled specifies whether the custom upstream cache is
|
||||||
|
// used. If true, the list of Upstreams should not be empty.
|
||||||
UpstreamsCacheEnabled bool
|
UpstreamsCacheEnabled bool
|
||||||
|
|
||||||
// UseOwnSettings specifies whether custom filtering settings are used.
|
// UseOwnSettings specifies whether custom filtering settings are used.
|
||||||
@@ -134,7 +136,7 @@ func (c *Persistent) validate(ctx context.Context, l *slog.Logger, allTags []str
|
|||||||
switch {
|
switch {
|
||||||
case c.Name == "":
|
case c.Name == "":
|
||||||
return errors.Error("empty name")
|
return errors.Error("empty name")
|
||||||
case c.IDsLen() == 0:
|
case c.idendifiersLen() == 0:
|
||||||
return errors.Error("id required")
|
return errors.Error("id required")
|
||||||
case c.UID == UID{}:
|
case c.UID == UID{}:
|
||||||
return errors.Error("uid required")
|
return errors.Error("uid required")
|
||||||
@@ -237,28 +239,15 @@ func (c *Persistent) setID(id string) (err error) {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
c.ClientIDs = append(c.ClientIDs, strings.ToLower(id))
|
c.ClientIDs = append(c.ClientIDs, ClientID(strings.ToLower(id)))
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// ValidateClientID returns an error if id is not a valid ClientID.
|
// Identifiers returns a list of client identifiers containing at least one
|
||||||
//
|
// element.
|
||||||
// TODO(s.chzhen): It's an exact copy of the [dnsforward.ValidateClientID] to
|
func (c *Persistent) Identifiers() (ids []string) {
|
||||||
// avoid the import cycle. Remove it.
|
ids = make([]string, 0, c.idendifiersLen())
|
||||||
func ValidateClientID(id string) (err error) {
|
|
||||||
err = netutil.ValidateHostnameLabel(id)
|
|
||||||
if err != nil {
|
|
||||||
// Replace the domain name label wrapper with our own.
|
|
||||||
return fmt.Errorf("invalid clientid %q: %w", id, errors.Unwrap(err))
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// IDs returns a list of ClientIDs containing at least one element.
|
|
||||||
func (c *Persistent) IDs() (ids []string) {
|
|
||||||
ids = make([]string, 0, c.IDsLen())
|
|
||||||
|
|
||||||
for _, ip := range c.IPs {
|
for _, ip := range c.IPs {
|
||||||
ids = append(ids, ip.String())
|
ids = append(ids, ip.String())
|
||||||
@@ -272,11 +261,15 @@ func (c *Persistent) IDs() (ids []string) {
|
|||||||
ids = append(ids, mac.String())
|
ids = append(ids, mac.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
return append(ids, c.ClientIDs...)
|
for _, cid := range c.ClientIDs {
|
||||||
|
ids = append(ids, string(cid))
|
||||||
|
}
|
||||||
|
|
||||||
|
return ids
|
||||||
}
|
}
|
||||||
|
|
||||||
// IDsLen returns a length of ClientIDs.
|
// identifiersLen returns the number of client identifiers.
|
||||||
func (c *Persistent) IDsLen() (n int) {
|
func (c *Persistent) idendifiersLen() (n int) {
|
||||||
return len(c.IPs) + len(c.Subnets) + len(c.MACs) + len(c.ClientIDs)
|
return len(c.IPs) + len(c.Subnets) + len(c.MACs) + len(c.ClientIDs)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import (
|
|||||||
"net"
|
"net"
|
||||||
"net/netip"
|
"net/netip"
|
||||||
"slices"
|
"slices"
|
||||||
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@@ -18,6 +19,7 @@ import (
|
|||||||
"github.com/AdguardTeam/golibs/errors"
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
"github.com/AdguardTeam/golibs/hostsfile"
|
"github.com/AdguardTeam/golibs/hostsfile"
|
||||||
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
||||||
|
"github.com/AdguardTeam/golibs/netutil"
|
||||||
"github.com/AdguardTeam/golibs/timeutil"
|
"github.com/AdguardTeam/golibs/timeutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -433,48 +435,186 @@ func (s *Storage) Add(ctx context.Context, p *Persistent) (err error) {
|
|||||||
ctx,
|
ctx,
|
||||||
"client added",
|
"client added",
|
||||||
"name", p.Name,
|
"name", p.Name,
|
||||||
"ids", p.IDs(),
|
"ids", p.Identifiers(),
|
||||||
"clients_count", s.index.size(),
|
"clients_count", s.index.size(),
|
||||||
)
|
)
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// FindByName finds persistent client by name. And returns its shallow copy.
|
// FindParams represents the parameters for searching a client. At least one
|
||||||
func (s *Storage) FindByName(name string) (p *Persistent, ok bool) {
|
// field must be non-empty.
|
||||||
s.mu.Lock()
|
type FindParams struct {
|
||||||
defer s.mu.Unlock()
|
// ClientID is a unique identifier for the client used in DoH, DoT, and DoQ
|
||||||
|
// DNS queries.
|
||||||
|
ClientID ClientID
|
||||||
|
|
||||||
p, ok = s.index.findByName(name)
|
// RemoteIP is the IP address used as a client search parameter.
|
||||||
if ok {
|
RemoteIP netip.Addr
|
||||||
return p.ShallowClone(), ok
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, false
|
// Subnet is the CIDR used as a client search parameter.
|
||||||
|
Subnet netip.Prefix
|
||||||
|
|
||||||
|
// MAC is the physical hardware address used as a client search parameter.
|
||||||
|
MAC net.HardwareAddr
|
||||||
|
|
||||||
|
// UID is the unique ID of persistent client used as a search parameter.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Use this.
|
||||||
|
UID UID
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find finds persistent client by string representation of the ClientID, IP
|
// ErrBadIdentifier is returned by [FindParams.Set] when it cannot parse the
|
||||||
// address, or MAC. And returns its shallow copy.
|
// provided client identifier.
|
||||||
|
const ErrBadIdentifier errors.Error = "bad client identifier"
|
||||||
|
|
||||||
|
// Set clears the stored search parameters and parses the string representation
|
||||||
|
// of the search parameter into typed parameter, storing it. In some cases, it
|
||||||
|
// may result in storing both an IP address and a MAC address because they might
|
||||||
|
// have identical string representations. It returns [ErrBadIdentifier] if id
|
||||||
|
// cannot be parsed.
|
||||||
//
|
//
|
||||||
// TODO(s.chzhen): Accept ClientIDData structure instead, which will contain
|
// TODO(s.chzhen): Add support for UID.
|
||||||
// the parsed IP address, if any.
|
func (p *FindParams) Set(id string) (err error) {
|
||||||
func (s *Storage) Find(id string) (p *Persistent, ok bool) {
|
*p = FindParams{}
|
||||||
|
|
||||||
|
isClientID := true
|
||||||
|
|
||||||
|
if netutil.IsValidIPString(id) {
|
||||||
|
// It is safe to use [netip.MustParseAddr] because it has already been
|
||||||
|
// validated that id contains the string representation of the IP
|
||||||
|
// address.
|
||||||
|
p.RemoteIP = netip.MustParseAddr(id)
|
||||||
|
|
||||||
|
// Even if id can be parsed as an IP address, it may be a MAC address.
|
||||||
|
// So do not return prematurely, continue parsing.
|
||||||
|
isClientID = false
|
||||||
|
}
|
||||||
|
|
||||||
|
if canBeValidIPPrefixString(id) {
|
||||||
|
p.Subnet, err = netip.ParsePrefix(id)
|
||||||
|
if err == nil {
|
||||||
|
isClientID = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if canBeMACString(id) {
|
||||||
|
p.MAC, err = net.ParseMAC(id)
|
||||||
|
if err == nil {
|
||||||
|
isClientID = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isClientID {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isValidClientID(id) {
|
||||||
|
return ErrBadIdentifier
|
||||||
|
}
|
||||||
|
|
||||||
|
p.ClientID = ClientID(id)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// canBeValidIPPrefixString is a best-effort check to determine if s is a valid
|
||||||
|
// CIDR before using [netip.ParsePrefix], aimed at reducing allocations.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Replace this implementation with the more robust version
|
||||||
|
// from golibs.
|
||||||
|
func canBeValidIPPrefixString(s string) (ok bool) {
|
||||||
|
ipStr, bitStr, ok := strings.Cut(s, "/")
|
||||||
|
if !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if bitStr == "" || len(bitStr) > 3 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
bits := 0
|
||||||
|
for _, c := range bitStr {
|
||||||
|
if c < '0' || c > '9' {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
bits = bits*10 + int(c-'0')
|
||||||
|
}
|
||||||
|
|
||||||
|
if bits > 128 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return netutil.IsValidIPString(ipStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// canBeMACString is a best-effort check to determine if s is a valid MAC
|
||||||
|
// address before using [net.ParseMAC], aimed at reducing allocations.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Replace this implementation with the more robust version
|
||||||
|
// from golibs.
|
||||||
|
func canBeMACString(s string) (ok bool) {
|
||||||
|
switch len(s) {
|
||||||
|
case
|
||||||
|
len("0000.0000.0000"),
|
||||||
|
len("00:00:00:00:00:00"),
|
||||||
|
len("0000.0000.0000.0000"),
|
||||||
|
len("00:00:00:00:00:00:00:00"),
|
||||||
|
len("0000.0000.0000.0000.0000.0000.0000.0000.0000.0000"),
|
||||||
|
len("00:00:00:00:00:00:00:00:00:00:00:00:00:00:00:00:00:00:00:00"):
|
||||||
|
return true
|
||||||
|
default:
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find represents the parameters for searching a client. params must not be
|
||||||
|
// nil and must have at least one non-empty field.
|
||||||
|
func (s *Storage) Find(params *FindParams) (p *Persistent, ok bool) {
|
||||||
s.mu.Lock()
|
s.mu.Lock()
|
||||||
defer s.mu.Unlock()
|
defer s.mu.Unlock()
|
||||||
|
|
||||||
p, ok = s.index.find(id)
|
isClientID := params.ClientID != ""
|
||||||
|
isRemoteIP := params.RemoteIP != (netip.Addr{})
|
||||||
|
isSubnet := params.Subnet != (netip.Prefix{})
|
||||||
|
isMAC := params.MAC != nil
|
||||||
|
|
||||||
|
for {
|
||||||
|
switch {
|
||||||
|
case isClientID:
|
||||||
|
isClientID = false
|
||||||
|
p, ok = s.index.findByClientID(params.ClientID)
|
||||||
|
case isRemoteIP:
|
||||||
|
isRemoteIP = false
|
||||||
|
p, ok = s.findByIP(params.RemoteIP)
|
||||||
|
case isSubnet:
|
||||||
|
isSubnet = false
|
||||||
|
p, ok = s.index.findByCIDR(params.Subnet)
|
||||||
|
case isMAC:
|
||||||
|
isMAC = false
|
||||||
|
p, ok = s.index.findByMAC(params.MAC)
|
||||||
|
default:
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
|
||||||
|
if ok {
|
||||||
|
return p.ShallowClone(), true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// findByIP finds persistent client by IP address. s.mu is expected to be
|
||||||
|
// locked.
|
||||||
|
func (s *Storage) findByIP(addr netip.Addr) (p *Persistent, ok bool) {
|
||||||
|
p, ok = s.index.findByIP(addr)
|
||||||
if ok {
|
if ok {
|
||||||
return p.ShallowClone(), ok
|
return p, true
|
||||||
}
|
}
|
||||||
|
|
||||||
ip, err := netip.ParseAddr(id)
|
foundMAC := s.dhcp.MACByIP(addr)
|
||||||
if err != nil {
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
foundMAC := s.dhcp.MACByIP(ip)
|
|
||||||
if foundMAC != nil {
|
if foundMAC != nil {
|
||||||
return s.FindByMAC(foundMAC)
|
return s.index.findByMAC(foundMAC)
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil, false
|
return nil, false
|
||||||
@@ -487,6 +627,8 @@ func (s *Storage) Find(id string) (p *Persistent, ok bool) {
|
|||||||
//
|
//
|
||||||
// Note that multiple clients can have the same IP address with different zones.
|
// Note that multiple clients can have the same IP address with different zones.
|
||||||
// Therefore, the result of this method is indeterminate.
|
// Therefore, the result of this method is indeterminate.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Consider accepting [FindParams].
|
||||||
func (s *Storage) FindLoose(ip netip.Addr, id string) (p *Persistent, ok bool) {
|
func (s *Storage) FindLoose(ip netip.Addr, id string) (p *Persistent, ok bool) {
|
||||||
s.mu.Lock()
|
s.mu.Lock()
|
||||||
defer s.mu.Unlock()
|
defer s.mu.Unlock()
|
||||||
@@ -498,7 +640,7 @@ func (s *Storage) FindLoose(ip netip.Addr, id string) (p *Persistent, ok bool) {
|
|||||||
|
|
||||||
foundMAC := s.dhcp.MACByIP(ip)
|
foundMAC := s.dhcp.MACByIP(ip)
|
||||||
if foundMAC != nil {
|
if foundMAC != nil {
|
||||||
return s.FindByMAC(foundMAC)
|
return s.index.findByMAC(foundMAC)
|
||||||
}
|
}
|
||||||
|
|
||||||
p = s.index.findByIPWithoutZone(ip)
|
p = s.index.findByIPWithoutZone(ip)
|
||||||
@@ -509,17 +651,6 @@ func (s *Storage) FindLoose(ip netip.Addr, id string) (p *Persistent, ok bool) {
|
|||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
// FindByMAC finds persistent client by MAC and returns its shallow copy. s.mu
|
|
||||||
// is expected to be locked.
|
|
||||||
func (s *Storage) FindByMAC(mac net.HardwareAddr) (p *Persistent, ok bool) {
|
|
||||||
p, ok = s.index.findByMAC(mac)
|
|
||||||
if ok {
|
|
||||||
return p.ShallowClone(), ok
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, false
|
|
||||||
}
|
|
||||||
|
|
||||||
// RemoveByName removes persistent client information. ok is false if no such
|
// RemoveByName removes persistent client information. ok is false if no such
|
||||||
// client exists by that name.
|
// client exists by that name.
|
||||||
func (s *Storage) RemoveByName(ctx context.Context, name string) (ok bool) {
|
func (s *Storage) RemoveByName(ctx context.Context, name string) (ok bool) {
|
||||||
@@ -648,9 +779,9 @@ func (s *Storage) CustomUpstreamConfig(
|
|||||||
s.mu.Lock()
|
s.mu.Lock()
|
||||||
defer s.mu.Unlock()
|
defer s.mu.Unlock()
|
||||||
|
|
||||||
c, ok := s.index.findByClientID(id)
|
c, ok := s.index.findByClientID(ClientID(id))
|
||||||
if !ok {
|
if !ok {
|
||||||
c, ok = s.index.findByIP(addr)
|
c, ok = s.findByIP(addr)
|
||||||
}
|
}
|
||||||
|
|
||||||
if !ok {
|
if !ok {
|
||||||
@@ -682,7 +813,7 @@ func (s *Storage) ClearUpstreamCache() {
|
|||||||
// ClientID or client IP address, and applies it to the filtering settings.
|
// ClientID or client IP address, and applies it to the filtering settings.
|
||||||
// setts must not be nil.
|
// setts must not be nil.
|
||||||
func (s *Storage) ApplyClientFiltering(id string, addr netip.Addr, setts *filtering.Settings) {
|
func (s *Storage) ApplyClientFiltering(id string, addr netip.Addr, setts *filtering.Settings) {
|
||||||
c, ok := s.index.findByClientID(id)
|
c, ok := s.index.findByClientID(ClientID(id))
|
||||||
if !ok {
|
if !ok {
|
||||||
c, ok = s.index.findByIP(addr)
|
c, ok = s.index.findByIP(addr)
|
||||||
}
|
}
|
||||||
@@ -690,7 +821,7 @@ func (s *Storage) ApplyClientFiltering(id string, addr netip.Addr, setts *filter
|
|||||||
if !ok {
|
if !ok {
|
||||||
foundMAC := s.dhcp.MACByIP(addr)
|
foundMAC := s.dhcp.MACByIP(addr)
|
||||||
if foundMAC != nil {
|
if foundMAC != nil {
|
||||||
c, ok = s.FindByMAC(foundMAC)
|
c, ok = s.index.findByMAC(foundMAC)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ import (
|
|||||||
"github.com/AdguardTeam/AdGuardHome/internal/dhcpsvc"
|
"github.com/AdguardTeam/AdGuardHome/internal/dhcpsvc"
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/dnsforward"
|
"github.com/AdguardTeam/AdGuardHome/internal/dnsforward"
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/whois"
|
"github.com/AdguardTeam/AdGuardHome/internal/whois"
|
||||||
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
"github.com/AdguardTeam/golibs/hostsfile"
|
"github.com/AdguardTeam/golibs/hostsfile"
|
||||||
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
||||||
"github.com/AdguardTeam/golibs/testutil"
|
"github.com/AdguardTeam/golibs/testutil"
|
||||||
@@ -350,15 +351,15 @@ func TestClientsDHCP(t *testing.T) {
|
|||||||
cliName1 = "one.dhcp"
|
cliName1 = "one.dhcp"
|
||||||
|
|
||||||
cliIP2 = netip.MustParseAddr("2.2.2.2")
|
cliIP2 = netip.MustParseAddr("2.2.2.2")
|
||||||
cliMAC2 = mustParseMAC("22:22:22:22:22:22")
|
cliMAC2 = errors.Must(net.ParseMAC("22:22:22:22:22:22"))
|
||||||
cliName2 = "two.dhcp"
|
cliName2 = "two.dhcp"
|
||||||
|
|
||||||
cliIP3 = netip.MustParseAddr("3.3.3.3")
|
cliIP3 = netip.MustParseAddr("3.3.3.3")
|
||||||
cliMAC3 = mustParseMAC("33:33:33:33:33:33")
|
cliMAC3 = errors.Must(net.ParseMAC("33:33:33:33:33:33"))
|
||||||
cliName3 = "three.dhcp"
|
cliName3 = "three.dhcp"
|
||||||
|
|
||||||
prsCliIP = netip.MustParseAddr("4.3.2.1")
|
prsCliIP = netip.MustParseAddr("4.3.2.1")
|
||||||
prsCliMAC = mustParseMAC("AA:AA:AA:AA:AA:AA")
|
prsCliMAC = errors.Must(net.ParseMAC("AA:AA:AA:AA:AA:AA"))
|
||||||
prsCliName = "persistent.dhcp"
|
prsCliName = "persistent.dhcp"
|
||||||
|
|
||||||
otherARPCliName = "other.arp"
|
otherARPCliName = "other.arp"
|
||||||
@@ -519,7 +520,11 @@ func TestClientsDHCP(t *testing.T) {
|
|||||||
})
|
})
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
prsCli, ok := storage.Find(prsCliIP.String())
|
params := &client.FindParams{}
|
||||||
|
err = params.Set(prsCliIP.String())
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
prsCli, ok := storage.Find(params)
|
||||||
require.True(t, ok)
|
require.True(t, ok)
|
||||||
|
|
||||||
assert.Equal(t, prsCliName, prsCli.Name)
|
assert.Equal(t, prsCliName, prsCli.Name)
|
||||||
@@ -663,17 +668,6 @@ func newStorage(tb testing.TB, m []*client.Persistent) (s *client.Storage) {
|
|||||||
return s
|
return s
|
||||||
}
|
}
|
||||||
|
|
||||||
// mustParseMAC is wrapper around [net.ParseMAC] that panics if there is an
|
|
||||||
// error.
|
|
||||||
func mustParseMAC(s string) (mac net.HardwareAddr) {
|
|
||||||
mac, err := net.ParseMAC(s)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return mac
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestStorage_Add(t *testing.T) {
|
func TestStorage_Add(t *testing.T) {
|
||||||
const (
|
const (
|
||||||
existingName = "existing_name"
|
existingName = "existing_name"
|
||||||
@@ -693,7 +687,7 @@ func TestStorage_Add(t *testing.T) {
|
|||||||
Name: existingName,
|
Name: existingName,
|
||||||
IPs: []netip.Addr{existingIP},
|
IPs: []netip.Addr{existingIP},
|
||||||
Subnets: []netip.Prefix{existingSubnet},
|
Subnets: []netip.Prefix{existingSubnet},
|
||||||
ClientIDs: []string{existingClientID},
|
ClientIDs: []client.ClientID{existingClientID},
|
||||||
UID: existingClientUID,
|
UID: existingClientUID,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -761,7 +755,7 @@ func TestStorage_Add(t *testing.T) {
|
|||||||
name: "duplicate_client_id",
|
name: "duplicate_client_id",
|
||||||
cli: &client.Persistent{
|
cli: &client.Persistent{
|
||||||
Name: "duplicate_client_id",
|
Name: "duplicate_client_id",
|
||||||
ClientIDs: []string{existingClientID},
|
ClientIDs: []client.ClientID{existingClientID},
|
||||||
UID: client.MustNewUID(),
|
UID: client.MustNewUID(),
|
||||||
},
|
},
|
||||||
wantErrMsg: `adding client: another client "existing_name" ` +
|
wantErrMsg: `adding client: another client "existing_name" ` +
|
||||||
@@ -898,12 +892,12 @@ func TestStorage_Find(t *testing.T) {
|
|||||||
|
|
||||||
clientWithMAC = &client.Persistent{
|
clientWithMAC = &client.Persistent{
|
||||||
Name: "client_with_mac",
|
Name: "client_with_mac",
|
||||||
MACs: []net.HardwareAddr{mustParseMAC(cliMAC)},
|
MACs: []net.HardwareAddr{errors.Must(net.ParseMAC(cliMAC))},
|
||||||
}
|
}
|
||||||
|
|
||||||
clientWithID = &client.Persistent{
|
clientWithID = &client.Persistent{
|
||||||
Name: "client_with_id",
|
Name: "client_with_id",
|
||||||
ClientIDs: []string{cliID},
|
ClientIDs: []client.ClientID{cliID},
|
||||||
}
|
}
|
||||||
|
|
||||||
clientLinkLocal = &client.Persistent{
|
clientLinkLocal = &client.Persistent{
|
||||||
@@ -950,7 +944,11 @@ func TestStorage_Find(t *testing.T) {
|
|||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
for _, id := range tc.ids {
|
for _, id := range tc.ids {
|
||||||
c, ok := s.Find(id)
|
params := &client.FindParams{}
|
||||||
|
err := params.Set(id)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
c, ok := s.Find(params)
|
||||||
require.True(t, ok)
|
require.True(t, ok)
|
||||||
|
|
||||||
assert.Equal(t, tc.want, c)
|
assert.Equal(t, tc.want, c)
|
||||||
@@ -959,7 +957,11 @@ func TestStorage_Find(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
t.Run("not_found", func(t *testing.T) {
|
t.Run("not_found", func(t *testing.T) {
|
||||||
_, ok := s.Find(cliIPNone)
|
params := &client.FindParams{}
|
||||||
|
err := params.Set(cliIPNone)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
_, ok := s.Find(params)
|
||||||
assert.False(t, ok)
|
assert.False(t, ok)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -1025,127 +1027,6 @@ func TestStorage_FindLoose(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestStorage_FindByName(t *testing.T) {
|
|
||||||
const (
|
|
||||||
cliIP1 = "1.1.1.1"
|
|
||||||
cliIP2 = "2.2.2.2"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
clientExistingName = "client_existing"
|
|
||||||
clientAnotherExistingName = "client_another_existing"
|
|
||||||
nonExistingClientName = "client_non_existing"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
clientExisting = &client.Persistent{
|
|
||||||
Name: clientExistingName,
|
|
||||||
IPs: []netip.Addr{netip.MustParseAddr(cliIP1)},
|
|
||||||
}
|
|
||||||
|
|
||||||
clientAnotherExisting = &client.Persistent{
|
|
||||||
Name: clientAnotherExistingName,
|
|
||||||
IPs: []netip.Addr{netip.MustParseAddr(cliIP2)},
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
clients := []*client.Persistent{
|
|
||||||
clientExisting,
|
|
||||||
clientAnotherExisting,
|
|
||||||
}
|
|
||||||
s := newStorage(t, clients)
|
|
||||||
|
|
||||||
testCases := []struct {
|
|
||||||
want *client.Persistent
|
|
||||||
name string
|
|
||||||
clientName string
|
|
||||||
}{{
|
|
||||||
name: "existing",
|
|
||||||
clientName: clientExistingName,
|
|
||||||
want: clientExisting,
|
|
||||||
}, {
|
|
||||||
name: "another_existing",
|
|
||||||
clientName: clientAnotherExistingName,
|
|
||||||
want: clientAnotherExisting,
|
|
||||||
}, {
|
|
||||||
name: "non_existing",
|
|
||||||
clientName: nonExistingClientName,
|
|
||||||
want: nil,
|
|
||||||
}}
|
|
||||||
|
|
||||||
for _, tc := range testCases {
|
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
|
||||||
c, ok := s.FindByName(tc.clientName)
|
|
||||||
if tc.want == nil {
|
|
||||||
assert.False(t, ok)
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.True(t, ok)
|
|
||||||
assert.Equal(t, tc.want, c)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestStorage_FindByMAC(t *testing.T) {
|
|
||||||
var (
|
|
||||||
cliMAC = mustParseMAC("11:11:11:11:11:11")
|
|
||||||
cliAnotherMAC = mustParseMAC("22:22:22:22:22:22")
|
|
||||||
nonExistingClientMAC = mustParseMAC("33:33:33:33:33:33")
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
clientExisting = &client.Persistent{
|
|
||||||
Name: "client",
|
|
||||||
MACs: []net.HardwareAddr{cliMAC},
|
|
||||||
}
|
|
||||||
|
|
||||||
clientAnotherExisting = &client.Persistent{
|
|
||||||
Name: "another_client",
|
|
||||||
MACs: []net.HardwareAddr{cliAnotherMAC},
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
clients := []*client.Persistent{
|
|
||||||
clientExisting,
|
|
||||||
clientAnotherExisting,
|
|
||||||
}
|
|
||||||
s := newStorage(t, clients)
|
|
||||||
|
|
||||||
testCases := []struct {
|
|
||||||
want *client.Persistent
|
|
||||||
name string
|
|
||||||
clientMAC net.HardwareAddr
|
|
||||||
}{{
|
|
||||||
name: "existing",
|
|
||||||
clientMAC: cliMAC,
|
|
||||||
want: clientExisting,
|
|
||||||
}, {
|
|
||||||
name: "another_existing",
|
|
||||||
clientMAC: cliAnotherMAC,
|
|
||||||
want: clientAnotherExisting,
|
|
||||||
}, {
|
|
||||||
name: "non_existing",
|
|
||||||
clientMAC: nonExistingClientMAC,
|
|
||||||
want: nil,
|
|
||||||
}}
|
|
||||||
|
|
||||||
for _, tc := range testCases {
|
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
|
||||||
c, ok := s.FindByMAC(tc.clientMAC)
|
|
||||||
if tc.want == nil {
|
|
||||||
assert.False(t, ok)
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.True(t, ok)
|
|
||||||
assert.Equal(t, tc.want, c)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestStorage_Update(t *testing.T) {
|
func TestStorage_Update(t *testing.T) {
|
||||||
const (
|
const (
|
||||||
clientName = "client_name"
|
clientName = "client_name"
|
||||||
@@ -1162,7 +1043,7 @@ func TestStorage_Update(t *testing.T) {
|
|||||||
Name: obstructingName,
|
Name: obstructingName,
|
||||||
IPs: []netip.Addr{obstructingIP},
|
IPs: []netip.Addr{obstructingIP},
|
||||||
Subnets: []netip.Prefix{obstructingSubnet},
|
Subnets: []netip.Prefix{obstructingSubnet},
|
||||||
ClientIDs: []string{obstructingClientID},
|
ClientIDs: []client.ClientID{obstructingClientID},
|
||||||
}
|
}
|
||||||
|
|
||||||
clientToUpdate := &client.Persistent{
|
clientToUpdate := &client.Persistent{
|
||||||
@@ -1211,7 +1092,7 @@ func TestStorage_Update(t *testing.T) {
|
|||||||
name: "duplicate_client_id",
|
name: "duplicate_client_id",
|
||||||
cli: &client.Persistent{
|
cli: &client.Persistent{
|
||||||
Name: "duplicate_client_id",
|
Name: "duplicate_client_id",
|
||||||
ClientIDs: []string{obstructingClientID},
|
ClientIDs: []client.ClientID{obstructingClientID},
|
||||||
UID: client.MustNewUID(),
|
UID: client.MustNewUID(),
|
||||||
},
|
},
|
||||||
wantErrMsg: `updating client: another client "obstructing_name" ` +
|
wantErrMsg: `updating client: another client "obstructing_name" ` +
|
||||||
@@ -1238,19 +1119,19 @@ func TestStorage_Update(t *testing.T) {
|
|||||||
func TestStorage_RangeByName(t *testing.T) {
|
func TestStorage_RangeByName(t *testing.T) {
|
||||||
sortedClients := []*client.Persistent{{
|
sortedClients := []*client.Persistent{{
|
||||||
Name: "clientA",
|
Name: "clientA",
|
||||||
ClientIDs: []string{"A"},
|
ClientIDs: []client.ClientID{"A"},
|
||||||
}, {
|
}, {
|
||||||
Name: "clientB",
|
Name: "clientB",
|
||||||
ClientIDs: []string{"B"},
|
ClientIDs: []client.ClientID{"B"},
|
||||||
}, {
|
}, {
|
||||||
Name: "clientC",
|
Name: "clientC",
|
||||||
ClientIDs: []string{"C"},
|
ClientIDs: []client.ClientID{"C"},
|
||||||
}, {
|
}, {
|
||||||
Name: "clientD",
|
Name: "clientD",
|
||||||
ClientIDs: []string{"D"},
|
ClientIDs: []client.ClientID{"D"},
|
||||||
}, {
|
}, {
|
||||||
Name: "clientE",
|
Name: "clientE",
|
||||||
ClientIDs: []string{"E"},
|
ClientIDs: []client.ClientID{"E"},
|
||||||
}}
|
}}
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
@@ -1288,29 +1169,20 @@ func TestStorage_RangeByName(t *testing.T) {
|
|||||||
|
|
||||||
func TestStorage_CustomUpstreamConfig(t *testing.T) {
|
func TestStorage_CustomUpstreamConfig(t *testing.T) {
|
||||||
const (
|
const (
|
||||||
existingName = "existing_name"
|
existingClientID = "existing_client_id"
|
||||||
existingClientID = "existing_client_id"
|
|
||||||
|
|
||||||
nonExistingClientID = "non_existing_client_id"
|
nonExistingClientID = "non_existing_client_id"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
existingClientUID = client.MustNewUID()
|
existingIP = netip.MustParseAddr("192.0.2.1")
|
||||||
existingIP = netip.MustParseAddr("192.0.2.1")
|
|
||||||
|
|
||||||
nonExistingIP = netip.MustParseAddr("192.0.2.255")
|
nonExistingIP = netip.MustParseAddr("192.0.2.255")
|
||||||
|
|
||||||
|
dhcpCliIP = netip.MustParseAddr("192.0.2.2")
|
||||||
|
dhcpCliMAC = errors.Must(net.ParseMAC("02:00:00:00:00:00"))
|
||||||
|
|
||||||
testUpstreamTimeout = time.Second
|
testUpstreamTimeout = time.Second
|
||||||
)
|
)
|
||||||
|
|
||||||
existingClient := &client.Persistent{
|
|
||||||
Name: existingName,
|
|
||||||
IPs: []netip.Addr{existingIP},
|
|
||||||
ClientIDs: []string{existingClientID},
|
|
||||||
UID: existingClientUID,
|
|
||||||
Upstreams: []string{"192.0.2.0"},
|
|
||||||
}
|
|
||||||
|
|
||||||
date := time.Now()
|
date := time.Now()
|
||||||
clock := &faketime.Clock{
|
clock := &faketime.Clock{
|
||||||
OnNow: func() (now time.Time) {
|
OnNow: func() (now time.Time) {
|
||||||
@@ -1320,7 +1192,30 @@ func TestStorage_CustomUpstreamConfig(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
s := newTestStorage(t, clock)
|
ipToMAC := map[netip.Addr]net.HardwareAddr{
|
||||||
|
dhcpCliIP: dhcpCliMAC,
|
||||||
|
}
|
||||||
|
|
||||||
|
dhcp := &testDHCP{
|
||||||
|
OnLeases: func() (ls []*dhcpsvc.Lease) {
|
||||||
|
panic("not implemented")
|
||||||
|
},
|
||||||
|
OnHostBy: func(ip netip.Addr) (host string) {
|
||||||
|
panic("not implemented")
|
||||||
|
},
|
||||||
|
OnMACBy: func(ip netip.Addr) (mac net.HardwareAddr) {
|
||||||
|
return ipToMAC[ip]
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx := testutil.ContextWithTimeout(t, testTimeout)
|
||||||
|
s, err := client.NewStorage(ctx, &client.StorageConfig{
|
||||||
|
Logger: slogutil.NewDiscardLogger(),
|
||||||
|
Clock: clock,
|
||||||
|
DHCP: dhcp,
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
s.UpdateCommonUpstreamConfig(&client.CommonUpstreamConfig{
|
s.UpdateCommonUpstreamConfig(&client.CommonUpstreamConfig{
|
||||||
UpstreamTimeout: testUpstreamTimeout,
|
UpstreamTimeout: testUpstreamTimeout,
|
||||||
})
|
})
|
||||||
@@ -1329,8 +1224,21 @@ func TestStorage_CustomUpstreamConfig(t *testing.T) {
|
|||||||
return s.Shutdown(testutil.ContextWithTimeout(t, testTimeout))
|
return s.Shutdown(testutil.ContextWithTimeout(t, testTimeout))
|
||||||
})
|
})
|
||||||
|
|
||||||
ctx := testutil.ContextWithTimeout(t, testTimeout)
|
err = s.Add(ctx, &client.Persistent{
|
||||||
err := s.Add(ctx, existingClient)
|
Name: "client_first",
|
||||||
|
IPs: []netip.Addr{existingIP},
|
||||||
|
ClientIDs: []client.ClientID{existingClientID},
|
||||||
|
UID: client.MustNewUID(),
|
||||||
|
Upstreams: []string{"192.0.2.0"},
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
err = s.Add(ctx, &client.Persistent{
|
||||||
|
Name: "client_second",
|
||||||
|
MACs: []net.HardwareAddr{dhcpCliMAC},
|
||||||
|
UID: client.MustNewUID(),
|
||||||
|
Upstreams: []string{"192.0.2.0"},
|
||||||
|
})
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
@@ -1348,6 +1256,11 @@ func TestStorage_CustomUpstreamConfig(t *testing.T) {
|
|||||||
cliID: "",
|
cliID: "",
|
||||||
cliAddr: existingIP,
|
cliAddr: existingIP,
|
||||||
wantNilConf: assert.NotNil,
|
wantNilConf: assert.NotNil,
|
||||||
|
}, {
|
||||||
|
name: "client_dhcp",
|
||||||
|
cliID: "",
|
||||||
|
cliAddr: dhcpCliIP,
|
||||||
|
wantNilConf: assert.NotNil,
|
||||||
}, {
|
}, {
|
||||||
name: "non_existing_client_id",
|
name: "non_existing_client_id",
|
||||||
cliID: nonExistingClientID,
|
cliID: nonExistingClientID,
|
||||||
@@ -1380,4 +1293,193 @@ func TestStorage_CustomUpstreamConfig(t *testing.T) {
|
|||||||
|
|
||||||
assert.NotEqual(t, conf, updConf)
|
assert.NotEqual(t, conf, updConf)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
t.Run("same_custom_config", func(t *testing.T) {
|
||||||
|
firstConf := s.CustomUpstreamConfig(existingClientID, existingIP)
|
||||||
|
require.NotNil(t, firstConf)
|
||||||
|
|
||||||
|
secondConf := s.CustomUpstreamConfig(existingClientID, existingIP)
|
||||||
|
require.NotNil(t, secondConf)
|
||||||
|
|
||||||
|
assert.Same(t, firstConf, secondConf)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkFindParams_Set(b *testing.B) {
|
||||||
|
const (
|
||||||
|
testIPStr = "192.0.2.1"
|
||||||
|
testCIDRStr = "192.0.2.0/24"
|
||||||
|
testMACStr = "02:00:00:00:00:00"
|
||||||
|
testClientID = "clientid"
|
||||||
|
)
|
||||||
|
|
||||||
|
benchCases := []struct {
|
||||||
|
wantErr error
|
||||||
|
params *client.FindParams
|
||||||
|
name string
|
||||||
|
id string
|
||||||
|
}{{
|
||||||
|
wantErr: nil,
|
||||||
|
params: &client.FindParams{
|
||||||
|
ClientID: testClientID,
|
||||||
|
},
|
||||||
|
name: "client_id",
|
||||||
|
id: testClientID,
|
||||||
|
}, {
|
||||||
|
wantErr: nil,
|
||||||
|
params: &client.FindParams{
|
||||||
|
RemoteIP: netip.MustParseAddr(testIPStr),
|
||||||
|
},
|
||||||
|
name: "ip_address",
|
||||||
|
id: testIPStr,
|
||||||
|
}, {
|
||||||
|
wantErr: nil,
|
||||||
|
params: &client.FindParams{
|
||||||
|
Subnet: netip.MustParsePrefix(testCIDRStr),
|
||||||
|
},
|
||||||
|
name: "subnet",
|
||||||
|
id: testCIDRStr,
|
||||||
|
}, {
|
||||||
|
wantErr: nil,
|
||||||
|
params: &client.FindParams{
|
||||||
|
MAC: errors.Must(net.ParseMAC(testMACStr)),
|
||||||
|
},
|
||||||
|
name: "mac_address",
|
||||||
|
id: testMACStr,
|
||||||
|
}, {
|
||||||
|
wantErr: client.ErrBadIdentifier,
|
||||||
|
params: &client.FindParams{},
|
||||||
|
name: "bad_id",
|
||||||
|
id: "!@#$%^&*()_+",
|
||||||
|
}}
|
||||||
|
|
||||||
|
for _, bc := range benchCases {
|
||||||
|
b.Run(bc.name, func(b *testing.B) {
|
||||||
|
params := &client.FindParams{}
|
||||||
|
var err error
|
||||||
|
|
||||||
|
b.ReportAllocs()
|
||||||
|
for b.Loop() {
|
||||||
|
err = params.Set(bc.id)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.ErrorIs(b, err, bc.wantErr)
|
||||||
|
assert.Equal(b, bc.params, params)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Most recent results:
|
||||||
|
//
|
||||||
|
// goos: linux
|
||||||
|
// goarch: amd64
|
||||||
|
// pkg: github.com/AdguardTeam/AdGuardHome/internal/client
|
||||||
|
// cpu: Intel(R) Core(TM) i7-10510U CPU @ 1.80GHz
|
||||||
|
// BenchmarkFindParams_Set/client_id-8 49463488 24.27 ns/op 0 B/op 0 allocs/op
|
||||||
|
// BenchmarkFindParams_Set/ip_address-8 18740977 62.22 ns/op 0 B/op 0 allocs/op
|
||||||
|
// BenchmarkFindParams_Set/subnet-8 10848192 110.0 ns/op 0 B/op 0 allocs/op
|
||||||
|
// BenchmarkFindParams_Set/mac_address-8 8148494 133.2 ns/op 8 B/op 1 allocs/op
|
||||||
|
// BenchmarkFindParams_Set/bad_id-8 73894278 16.29 ns/op 0 B/op 0 allocs/op
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkStorage_Find(b *testing.B) {
|
||||||
|
const (
|
||||||
|
cliID = "cid"
|
||||||
|
cliMAC = "02:00:00:00:00:00"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
cliNameWithID = "client_with_id"
|
||||||
|
cliNameWithIP = "client_with_ip"
|
||||||
|
cliNameWithCIDR = "client_with_cidr"
|
||||||
|
cliNameWithMAC = "client_with_mac"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
cliIP = netip.MustParseAddr("192.0.2.1")
|
||||||
|
cliCIDR = netip.MustParsePrefix("192.0.2.0/24")
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
clientWithID = &client.Persistent{
|
||||||
|
Name: cliNameWithID,
|
||||||
|
ClientIDs: []client.ClientID{cliID},
|
||||||
|
}
|
||||||
|
clientWithIP = &client.Persistent{
|
||||||
|
Name: cliNameWithIP,
|
||||||
|
IPs: []netip.Addr{cliIP},
|
||||||
|
}
|
||||||
|
clientWithCIDR = &client.Persistent{
|
||||||
|
Name: cliNameWithCIDR,
|
||||||
|
Subnets: []netip.Prefix{cliCIDR},
|
||||||
|
}
|
||||||
|
clientWithMAC = &client.Persistent{
|
||||||
|
Name: cliNameWithMAC,
|
||||||
|
MACs: []net.HardwareAddr{errors.Must(net.ParseMAC(cliMAC))},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
clients := []*client.Persistent{
|
||||||
|
clientWithID,
|
||||||
|
clientWithIP,
|
||||||
|
clientWithCIDR,
|
||||||
|
clientWithMAC,
|
||||||
|
}
|
||||||
|
s := newStorage(b, clients)
|
||||||
|
|
||||||
|
benchCases := []struct {
|
||||||
|
params *client.FindParams
|
||||||
|
name string
|
||||||
|
wantName string
|
||||||
|
}{{
|
||||||
|
params: &client.FindParams{
|
||||||
|
ClientID: cliID,
|
||||||
|
},
|
||||||
|
name: "client_id",
|
||||||
|
wantName: cliNameWithID,
|
||||||
|
}, {
|
||||||
|
params: &client.FindParams{
|
||||||
|
RemoteIP: cliIP,
|
||||||
|
},
|
||||||
|
name: "ip_address",
|
||||||
|
wantName: cliNameWithIP,
|
||||||
|
}, {
|
||||||
|
params: &client.FindParams{
|
||||||
|
Subnet: cliCIDR,
|
||||||
|
},
|
||||||
|
name: "subnet",
|
||||||
|
wantName: cliNameWithCIDR,
|
||||||
|
}, {
|
||||||
|
params: &client.FindParams{
|
||||||
|
MAC: errors.Must(net.ParseMAC(cliMAC)),
|
||||||
|
},
|
||||||
|
name: "mac_address",
|
||||||
|
wantName: cliNameWithMAC,
|
||||||
|
}}
|
||||||
|
|
||||||
|
for _, bc := range benchCases {
|
||||||
|
b.Run(bc.name, func(b *testing.B) {
|
||||||
|
var p *client.Persistent
|
||||||
|
var ok bool
|
||||||
|
|
||||||
|
b.ReportAllocs()
|
||||||
|
for b.Loop() {
|
||||||
|
p, ok = s.Find(bc.params)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.True(b, ok)
|
||||||
|
assert.NotNil(b, p)
|
||||||
|
assert.Equal(b, bc.wantName, p.Name)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Most recent results:
|
||||||
|
//
|
||||||
|
// goos: linux
|
||||||
|
// goarch: amd64
|
||||||
|
// pkg: github.com/AdguardTeam/AdGuardHome/internal/client
|
||||||
|
// cpu: Intel(R) Core(TM) i7-10510U CPU @ 1.80GHz
|
||||||
|
// BenchmarkStorage_Find/client_id-8 7070107 154.4 ns/op 240 B/op 2 allocs/op
|
||||||
|
// BenchmarkStorage_Find/ip_address-8 6831823 168.6 ns/op 248 B/op 2 allocs/op
|
||||||
|
// BenchmarkStorage_Find/subnet-8 7209050 167.5 ns/op 256 B/op 2 allocs/op
|
||||||
|
// BenchmarkStorage_Find/mac_address-8 5776131 199.7 ns/op 256 B/op 3 allocs/op
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -138,6 +138,7 @@ func (m *upstreamManager) customUpstreamConfig(uid UID) (proxyConf *proxy.Custom
|
|||||||
|
|
||||||
proxyConf = newCustomUpstreamConfig(cliConf, m.commonConf)
|
proxyConf = newCustomUpstreamConfig(cliConf, m.commonConf)
|
||||||
cliConf.proxyConf = proxyConf
|
cliConf.proxyConf = proxyConf
|
||||||
|
cliConf.commonConfUpdate = m.confUpdate
|
||||||
cliConf.isChanged = false
|
cliConf.isChanged = false
|
||||||
|
|
||||||
return proxyConf
|
return proxyConf
|
||||||
|
|||||||
@@ -1,13 +1,11 @@
|
|||||||
package dhcpsvc_test
|
package dhcpsvc_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net"
|
|
||||||
"net/netip"
|
"net/netip"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/dhcpsvc"
|
"github.com/AdguardTeam/AdGuardHome/internal/dhcpsvc"
|
||||||
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
||||||
"github.com/stretchr/testify/require"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// testLocalTLD is a common local TLD for tests.
|
// testLocalTLD is a common local TLD for tests.
|
||||||
@@ -56,11 +54,3 @@ var testInterfaceConf = map[string]*dhcpsvc.InterfaceConfig{
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
// mustParseMAC parses a hardware address from s and requires no errors.
|
|
||||||
func mustParseMAC(t require.TestingT, s string) (mac net.HardwareAddr) {
|
|
||||||
mac, err := net.ParseMAC(s)
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
return mac
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package dhcpsvc_test
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"io/fs"
|
"io/fs"
|
||||||
|
"net"
|
||||||
"net/netip"
|
"net/netip"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
@@ -11,6 +12,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/dhcpsvc"
|
"github.com/AdguardTeam/AdGuardHome/internal/dhcpsvc"
|
||||||
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
"github.com/AdguardTeam/golibs/testutil"
|
"github.com/AdguardTeam/golibs/testutil"
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
@@ -176,9 +178,9 @@ func TestDHCPServer_AddLease(t *testing.T) {
|
|||||||
newIP = netip.MustParseAddr("192.168.0.3")
|
newIP = netip.MustParseAddr("192.168.0.3")
|
||||||
newIPv6 = netip.MustParseAddr("2001:db8::2")
|
newIPv6 = netip.MustParseAddr("2001:db8::2")
|
||||||
|
|
||||||
existMAC = mustParseMAC(t, "01:02:03:04:05:06")
|
existMAC = errors.Must(net.ParseMAC("01:02:03:04:05:06"))
|
||||||
newMAC = mustParseMAC(t, "06:05:04:03:02:01")
|
newMAC = errors.Must(net.ParseMAC("06:05:04:03:02:01"))
|
||||||
ipv6MAC = mustParseMAC(t, "02:03:04:05:06:07")
|
ipv6MAC = errors.Must(net.ParseMAC("02:03:04:05:06:07"))
|
||||||
)
|
)
|
||||||
|
|
||||||
require.NoError(t, srv.AddLease(ctx, &dhcpsvc.Lease{
|
require.NoError(t, srv.AddLease(ctx, &dhcpsvc.Lease{
|
||||||
@@ -291,9 +293,9 @@ func TestDHCPServer_index(t *testing.T) {
|
|||||||
ip3 = netip.MustParseAddr("172.16.0.3")
|
ip3 = netip.MustParseAddr("172.16.0.3")
|
||||||
ip4 = netip.MustParseAddr("172.16.0.4")
|
ip4 = netip.MustParseAddr("172.16.0.4")
|
||||||
|
|
||||||
mac1 = mustParseMAC(t, "01:02:03:04:05:06")
|
mac1 = errors.Must(net.ParseMAC("01:02:03:04:05:06"))
|
||||||
mac2 = mustParseMAC(t, "06:05:04:03:02:01")
|
mac2 = errors.Must(net.ParseMAC("06:05:04:03:02:01"))
|
||||||
mac3 = mustParseMAC(t, "02:03:04:05:06:07")
|
mac3 = errors.Must(net.ParseMAC("02:03:04:05:06:07"))
|
||||||
)
|
)
|
||||||
|
|
||||||
t.Run("ip_idx", func(t *testing.T) {
|
t.Run("ip_idx", func(t *testing.T) {
|
||||||
@@ -349,9 +351,9 @@ func TestDHCPServer_UpdateStaticLease(t *testing.T) {
|
|||||||
ip3 = netip.MustParseAddr("192.168.0.4")
|
ip3 = netip.MustParseAddr("192.168.0.4")
|
||||||
ip4 = netip.MustParseAddr("2001:db8::3")
|
ip4 = netip.MustParseAddr("2001:db8::3")
|
||||||
|
|
||||||
mac1 = mustParseMAC(t, "01:02:03:04:05:06")
|
mac1 = errors.Must(net.ParseMAC("01:02:03:04:05:06"))
|
||||||
mac2 = mustParseMAC(t, "06:05:04:03:02:01")
|
mac2 = errors.Must(net.ParseMAC("06:05:04:03:02:01"))
|
||||||
mac3 = mustParseMAC(t, "06:05:04:03:02:02")
|
mac3 = errors.Must(net.ParseMAC("06:05:04:03:02:02"))
|
||||||
)
|
)
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
@@ -452,9 +454,9 @@ func TestDHCPServer_RemoveLease(t *testing.T) {
|
|||||||
newIP = netip.MustParseAddr("192.168.0.3")
|
newIP = netip.MustParseAddr("192.168.0.3")
|
||||||
newIPv6 = netip.MustParseAddr("2001:db8::2")
|
newIPv6 = netip.MustParseAddr("2001:db8::2")
|
||||||
|
|
||||||
existMAC = mustParseMAC(t, "01:02:03:04:05:06")
|
existMAC = errors.Must(net.ParseMAC("01:02:03:04:05:06"))
|
||||||
newMAC = mustParseMAC(t, "02:03:04:05:06:07")
|
newMAC = errors.Must(net.ParseMAC("02:03:04:05:06:07"))
|
||||||
ipv6MAC = mustParseMAC(t, "06:05:04:03:02:01")
|
ipv6MAC = errors.Must(net.ParseMAC("06:05:04:03:02:01"))
|
||||||
)
|
)
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
@@ -559,13 +561,13 @@ func TestServer_Leases(t *testing.T) {
|
|||||||
Expiry: expiry,
|
Expiry: expiry,
|
||||||
IP: netip.MustParseAddr("192.168.0.3"),
|
IP: netip.MustParseAddr("192.168.0.3"),
|
||||||
Hostname: "example.host",
|
Hostname: "example.host",
|
||||||
HWAddr: mustParseMAC(t, "AA:AA:AA:AA:AA:AA"),
|
HWAddr: errors.Must(net.ParseMAC("AA:AA:AA:AA:AA:AA")),
|
||||||
IsStatic: false,
|
IsStatic: false,
|
||||||
}, {
|
}, {
|
||||||
Expiry: time.Time{},
|
Expiry: time.Time{},
|
||||||
IP: netip.MustParseAddr("192.168.0.4"),
|
IP: netip.MustParseAddr("192.168.0.4"),
|
||||||
Hostname: "example.static.host",
|
Hostname: "example.static.host",
|
||||||
HWAddr: mustParseMAC(t, "BB:BB:BB:BB:BB:BB"),
|
HWAddr: errors.Must(net.ParseMAC("BB:BB:BB:BB:BB:BB")),
|
||||||
IsStatic: true,
|
IsStatic: true,
|
||||||
}}
|
}}
|
||||||
assert.ElementsMatch(t, wantLeases, srv.Leases())
|
assert.ElementsMatch(t, wantLeases, srv.Leases())
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import (
|
|||||||
|
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghalg"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghalg"
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
|
||||||
|
"github.com/AdguardTeam/AdGuardHome/internal/client"
|
||||||
"github.com/AdguardTeam/golibs/container"
|
"github.com/AdguardTeam/golibs/container"
|
||||||
"github.com/AdguardTeam/golibs/log"
|
"github.com/AdguardTeam/golibs/log"
|
||||||
"github.com/AdguardTeam/golibs/stringutil"
|
"github.com/AdguardTeam/golibs/stringutil"
|
||||||
@@ -51,7 +52,7 @@ func processAccessClients(
|
|||||||
} else if ipnet, err = netip.ParsePrefix(s); err == nil {
|
} else if ipnet, err = netip.ParsePrefix(s); err == nil {
|
||||||
*nets = append(*nets, ipnet)
|
*nets = append(*nets, ipnet)
|
||||||
} else {
|
} else {
|
||||||
err = ValidateClientID(s)
|
err = client.ValidateClientID(s)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("value %q at index %d: bad ip, cidr, or clientid", s, i)
|
return fmt.Errorf("value %q at index %d: bad ip, cidr, or clientid", s, i)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -74,7 +74,7 @@ func (s *Server) clientIDFromDNSContext(pctx *proxy.DNSContext) (clientID string
|
|||||||
return "", nil
|
return "", nil
|
||||||
}
|
}
|
||||||
|
|
||||||
hostSrvName := s.conf.ServerName
|
hostSrvName := s.conf.TLSConf.ServerName
|
||||||
if hostSrvName == "" {
|
if hostSrvName == "" {
|
||||||
return "", nil
|
return "", nil
|
||||||
}
|
}
|
||||||
@@ -87,7 +87,7 @@ func (s *Server) clientIDFromDNSContext(pctx *proxy.DNSContext) (clientID string
|
|||||||
clientID, err = clientIDFromClientServerName(
|
clientID, err = clientIDFromClientServerName(
|
||||||
hostSrvName,
|
hostSrvName,
|
||||||
cliSrvName,
|
cliSrvName,
|
||||||
s.conf.StrictSNICheck,
|
s.conf.TLSConf.StrictSNICheck,
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", fmt.Errorf("clientid check: %w", err)
|
return "", fmt.Errorf("clientid check: %w", err)
|
||||||
|
|||||||
@@ -121,7 +121,7 @@ func TestServer_HandleBefore_tls(t *testing.T) {
|
|||||||
t.Run(tc.name, func(t *testing.T) {
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
s, _ := createTestTLS(t, TLSConfig{
|
s, _ := createTestTLS(t, &TLSConfig{
|
||||||
TLSListenAddrs: []*net.TCPAddr{{}},
|
TLSListenAddrs: []*net.TCPAddr{{}},
|
||||||
ServerName: tlsServerName,
|
ServerName: tlsServerName,
|
||||||
})
|
})
|
||||||
@@ -259,6 +259,7 @@ func TestServer_HandleBefore_udp(t *testing.T) {
|
|||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
AllowedClients: tc.allowedClients,
|
AllowedClients: tc.allowedClients,
|
||||||
DisallowedClients: tc.disallowedClients,
|
DisallowedClients: tc.disallowedClients,
|
||||||
|
|||||||
@@ -7,26 +7,13 @@ import (
|
|||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/AdguardTeam/AdGuardHome/internal/client"
|
||||||
"github.com/AdguardTeam/dnsproxy/proxy"
|
"github.com/AdguardTeam/dnsproxy/proxy"
|
||||||
"github.com/AdguardTeam/golibs/errors"
|
|
||||||
"github.com/AdguardTeam/golibs/log"
|
"github.com/AdguardTeam/golibs/log"
|
||||||
"github.com/AdguardTeam/golibs/netutil"
|
"github.com/AdguardTeam/golibs/netutil"
|
||||||
"github.com/quic-go/quic-go"
|
"github.com/quic-go/quic-go"
|
||||||
)
|
)
|
||||||
|
|
||||||
// ValidateClientID returns an error if id is not a valid ClientID.
|
|
||||||
//
|
|
||||||
// Keep in sync with [client.ValidateClientID].
|
|
||||||
func ValidateClientID(id string) (err error) {
|
|
||||||
err = netutil.ValidateHostnameLabel(id)
|
|
||||||
if err != nil {
|
|
||||||
// Replace the domain name label wrapper with our own.
|
|
||||||
return fmt.Errorf("invalid clientid %q: %w", id, errors.Unwrap(err))
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// clientIDFromClientServerName extracts and validates a ClientID. hostSrvName
|
// clientIDFromClientServerName extracts and validates a ClientID. hostSrvName
|
||||||
// is the server name of the host. cliSrvName is the server name as sent by the
|
// is the server name of the host. cliSrvName is the server name as sent by the
|
||||||
// client. When strict is true, and client and host server name don't match,
|
// client. When strict is true, and client and host server name don't match,
|
||||||
@@ -53,7 +40,7 @@ func clientIDFromClientServerName(
|
|||||||
}
|
}
|
||||||
|
|
||||||
clientID = cliSrvName[:len(cliSrvName)-len(hostSrvName)-1]
|
clientID = cliSrvName[:len(cliSrvName)-len(hostSrvName)-1]
|
||||||
err = ValidateClientID(clientID)
|
err = client.ValidateClientID(clientID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// Don't wrap the error, because it's informative enough as is.
|
// Don't wrap the error, because it's informative enough as is.
|
||||||
return "", err
|
return "", err
|
||||||
@@ -93,7 +80,7 @@ func clientIDFromDNSContextHTTPS(pctx *proxy.DNSContext) (clientID string, err e
|
|||||||
return "", fmt.Errorf("clientid check: invalid path %q: extra parts", origPath)
|
return "", fmt.Errorf("clientid check: invalid path %q: extra parts", origPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
err = ValidateClientID(clientID)
|
err = client.ValidateClientID(clientID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", fmt.Errorf("clientid check: %w", err)
|
return "", fmt.Errorf("clientid check: %w", err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -212,13 +212,13 @@ func TestServer_clientIDFromDNSContext(t *testing.T) {
|
|||||||
|
|
||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
tlsConf := TLSConfig{
|
tlsConf := &TLSConfig{
|
||||||
ServerName: tc.confSrvName,
|
ServerName: tc.confSrvName,
|
||||||
StrictSNICheck: tc.strictSNI,
|
StrictSNICheck: tc.strictSNI,
|
||||||
}
|
}
|
||||||
|
|
||||||
srv := &Server{
|
srv := &Server{
|
||||||
conf: ServerConfig{TLSConfig: tlsConf},
|
conf: ServerConfig{TLSConf: tlsConf},
|
||||||
baseLogger: slogutil.NewDiscardLogger(),
|
baseLogger: slogutil.NewDiscardLogger(),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -11,7 +11,6 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghalg"
|
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghnet"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghnet"
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghtls"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghtls"
|
||||||
@@ -168,43 +167,34 @@ type EDNSClientSubnet struct {
|
|||||||
UseCustom bool `yaml:"use_custom"`
|
UseCustom bool `yaml:"use_custom"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// TLSConfig is the TLS configuration for HTTPS, DNS-over-HTTPS, and DNS-over-TLS
|
// TLSConfig contains the TLS configuration settings for DNS-over-HTTPS (DoH),
|
||||||
|
// DNS-over-TLS (DoT), DNS-over-QUIC (DoQ), and Discovery of Designated
|
||||||
|
// Resolvers (DDR).
|
||||||
type TLSConfig struct {
|
type TLSConfig struct {
|
||||||
cert tls.Certificate
|
// Cert is the TLS certificate used for TLS connections. It is nil if
|
||||||
|
// encryption is disabled.
|
||||||
|
Cert *tls.Certificate
|
||||||
|
|
||||||
TLSListenAddrs []*net.TCPAddr `yaml:"-" json:"-"`
|
// TLSListenAddrs are the addresses to listen on for DoT connections. Each
|
||||||
QUICListenAddrs []*net.UDPAddr `yaml:"-" json:"-"`
|
// item in the list must be non-nil if Cert is not nil.
|
||||||
HTTPSListenAddrs []*net.TCPAddr `yaml:"-" json:"-"`
|
TLSListenAddrs []*net.TCPAddr
|
||||||
|
|
||||||
// PEM-encoded certificates chain
|
// QUICListenAddrs are the addresses to listen on for DoQ connections. Each
|
||||||
CertificateChain string `yaml:"certificate_chain" json:"certificate_chain"`
|
// item in the list must be non-nil if Cert is not nil.
|
||||||
// PEM-encoded private key
|
QUICListenAddrs []*net.UDPAddr
|
||||||
PrivateKey string `yaml:"private_key" json:"private_key"`
|
|
||||||
|
|
||||||
CertificatePath string `yaml:"certificate_path" json:"certificate_path"`
|
// HTTPSListenAddrs should be the addresses AdGuard Home is listening on for
|
||||||
PrivateKeyPath string `yaml:"private_key_path" json:"private_key_path"`
|
// DoH connections. These addresses are announced with DDR. Each item in
|
||||||
|
// the list must be non-nil.
|
||||||
CertificateChainData []byte `yaml:"-" json:"-"`
|
HTTPSListenAddrs []*net.TCPAddr
|
||||||
PrivateKeyData []byte `yaml:"-" json:"-"`
|
|
||||||
|
|
||||||
// ServerName is the hostname of the server. Currently, it is only being
|
// ServerName is the hostname of the server. Currently, it is only being
|
||||||
// used for ClientID checking and Discovery of Designated Resolvers (DDR).
|
// used for ClientID checking and Discovery of Designated Resolvers (DDR).
|
||||||
ServerName string `yaml:"-" json:"-"`
|
ServerName string
|
||||||
|
|
||||||
// DNS names from certificate (SAN) or CN value from Subject
|
|
||||||
dnsNames []string
|
|
||||||
|
|
||||||
// OverrideTLSCiphers, when set, contains the names of the cipher suites to
|
|
||||||
// use. If the slice is empty, the default safe suites are used.
|
|
||||||
OverrideTLSCiphers []string `yaml:"override_tls_ciphers,omitempty" json:"-"`
|
|
||||||
|
|
||||||
// StrictSNICheck controls if the connections with SNI mismatching the
|
// StrictSNICheck controls if the connections with SNI mismatching the
|
||||||
// certificate's ones should be rejected.
|
// certificate's ones should be rejected.
|
||||||
StrictSNICheck bool `yaml:"strict_sni_check" json:"-"`
|
StrictSNICheck bool
|
||||||
|
|
||||||
// hasIPAddrs is set during the certificate parsing and is true if the
|
|
||||||
// configured certificate contains at least a single IP address.
|
|
||||||
hasIPAddrs bool
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// DNSCryptConfig is the DNSCrypt server configuration struct.
|
// DNSCryptConfig is the DNSCrypt server configuration struct.
|
||||||
@@ -239,8 +229,11 @@ type ServerConfig struct {
|
|||||||
// Remove that.
|
// Remove that.
|
||||||
AddrProcConf *client.DefaultAddrProcConfig
|
AddrProcConf *client.DefaultAddrProcConfig
|
||||||
|
|
||||||
|
// TLSConf is the TLS configuration for DNS-over-TLS, DNS-over-QUIC, and
|
||||||
|
// HTTPS. It must not be nil.
|
||||||
|
TLSConf *TLSConfig
|
||||||
|
|
||||||
Config
|
Config
|
||||||
TLSConfig
|
|
||||||
DNSCryptConfig
|
DNSCryptConfig
|
||||||
TLSAllowUnencryptedDoH bool
|
TLSAllowUnencryptedDoH bool
|
||||||
|
|
||||||
@@ -281,6 +274,10 @@ type ServerConfig struct {
|
|||||||
|
|
||||||
// ServePlainDNS defines if plain DNS is allowed for incoming requests.
|
// ServePlainDNS defines if plain DNS is allowed for incoming requests.
|
||||||
ServePlainDNS bool
|
ServePlainDNS bool
|
||||||
|
|
||||||
|
// PendingRequestsEnabled defines if duplicate requests should be forwarded
|
||||||
|
// to upstreams along with the original one.
|
||||||
|
PendingRequestsEnabled bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpstreamMode is a enumeration of upstream mode representations. See
|
// UpstreamMode is a enumeration of upstream mode representations. See
|
||||||
@@ -324,6 +321,9 @@ func (s *Server) newProxyConfig() (conf *proxy.Config, err error) {
|
|||||||
UsePrivateRDNS: srvConf.UsePrivateRDNS,
|
UsePrivateRDNS: srvConf.UsePrivateRDNS,
|
||||||
PrivateSubnets: s.privateNets,
|
PrivateSubnets: s.privateNets,
|
||||||
MessageConstructor: s,
|
MessageConstructor: s,
|
||||||
|
PendingRequests: &proxy.PendingRequestsConfig{
|
||||||
|
Enabled: srvConf.PendingRequestsEnabled,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
if srvConf.EDNSClientSubnet.UseCustom {
|
if srvConf.EDNSClientSubnet.UseCustom {
|
||||||
@@ -608,45 +608,33 @@ func (conf *ServerConfig) ourAddrsSet() (m addrPortSet, err error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// prepareTLS - prepares TLS configuration for the DNS proxy
|
// prepareTLS sets up the TLS configuration for the DNS proxy.
|
||||||
func (s *Server) prepareTLS(proxyConfig *proxy.Config) (err error) {
|
func (s *Server) prepareTLS(proxyConfig *proxy.Config) (err error) {
|
||||||
if len(s.conf.CertificateChainData) == 0 || len(s.conf.PrivateKeyData) == 0 {
|
if s.conf.TLSConf.Cert == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if s.conf.TLSConf.TLSListenAddrs == nil && s.conf.TLSConf.QUICListenAddrs == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if s.conf.TLSListenAddrs == nil && s.conf.QUICListenAddrs == nil {
|
proxyConfig.TLSListenAddr = s.conf.TLSConf.TLSListenAddrs
|
||||||
return nil
|
proxyConfig.QUICListenAddr = s.conf.TLSConf.QUICListenAddrs
|
||||||
}
|
|
||||||
|
|
||||||
proxyConfig.TLSListenAddr = aghalg.CoalesceSlice(
|
cert, err := x509.ParseCertificate(s.conf.TLSConf.Cert.Certificate[0])
|
||||||
s.conf.TLSListenAddrs,
|
|
||||||
proxyConfig.TLSListenAddr,
|
|
||||||
)
|
|
||||||
|
|
||||||
proxyConfig.QUICListenAddr = aghalg.CoalesceSlice(
|
|
||||||
s.conf.QUICListenAddrs,
|
|
||||||
proxyConfig.QUICListenAddr,
|
|
||||||
)
|
|
||||||
|
|
||||||
s.conf.cert, err = tls.X509KeyPair(s.conf.CertificateChainData, s.conf.PrivateKeyData)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("failed to parse TLS keypair: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
cert, err := x509.ParseCertificate(s.conf.cert.Certificate[0])
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("x509.ParseCertificate(): %w", err)
|
return fmt.Errorf("x509.ParseCertificate(): %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
s.conf.hasIPAddrs = aghtls.CertificateHasIP(cert)
|
s.hasIPAddrs = aghtls.CertificateHasIP(cert)
|
||||||
|
|
||||||
if s.conf.StrictSNICheck {
|
if s.conf.TLSConf.StrictSNICheck {
|
||||||
if len(cert.DNSNames) != 0 {
|
if len(cert.DNSNames) != 0 {
|
||||||
s.conf.dnsNames = cert.DNSNames
|
s.dnsNames = cert.DNSNames
|
||||||
log.Debug("dns: using certificate's SAN as DNS names: %v", cert.DNSNames)
|
log.Debug("dns: using certificate's SAN as DNS names: %v", cert.DNSNames)
|
||||||
slices.Sort(s.conf.dnsNames)
|
slices.Sort(s.dnsNames)
|
||||||
} else {
|
} else {
|
||||||
s.conf.dnsNames = append(s.conf.dnsNames, cert.Subject.CommonName)
|
s.dnsNames = []string{cert.Subject.CommonName}
|
||||||
log.Debug("dns: using certificate's CN as DNS name: %s", cert.Subject.CommonName)
|
log.Debug("dns: using certificate's CN as DNS name: %s", cert.Subject.CommonName)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -695,11 +683,11 @@ func anyNameMatches(dnsNames []string, sni string) (ok bool) {
|
|||||||
// Called by 'tls' package when Client Hello is received
|
// Called by 'tls' package when Client Hello is received
|
||||||
// If the server name (from SNI) supplied by client is incorrect - we terminate the ongoing TLS handshake.
|
// If the server name (from SNI) supplied by client is incorrect - we terminate the ongoing TLS handshake.
|
||||||
func (s *Server) onGetCertificate(ch *tls.ClientHelloInfo) (*tls.Certificate, error) {
|
func (s *Server) onGetCertificate(ch *tls.ClientHelloInfo) (*tls.Certificate, error) {
|
||||||
if s.conf.StrictSNICheck && !anyNameMatches(s.conf.dnsNames, ch.ServerName) {
|
if s.conf.TLSConf.StrictSNICheck && !anyNameMatches(s.dnsNames, ch.ServerName) {
|
||||||
log.Info("dns: tls: unknown SNI in Client Hello: %s", ch.ServerName)
|
log.Info("dns: tls: unknown SNI in Client Hello: %s", ch.ServerName)
|
||||||
return nil, fmt.Errorf("invalid SNI")
|
return nil, fmt.Errorf("invalid SNI")
|
||||||
}
|
}
|
||||||
return &s.conf.cert, nil
|
return s.conf.TLSConf.Cert, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// preparePlain prepares the plain-DNS configuration for the DNS proxy.
|
// preparePlain prepares the plain-DNS configuration for the DNS proxy.
|
||||||
|
|||||||
@@ -296,6 +296,7 @@ func TestServer_HandleDNSRequest_dns64(t *testing.T) {
|
|||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
UseDNS64: true,
|
UseDNS64: true,
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
@@ -335,6 +336,7 @@ func TestServer_dns64WithDisabledRDNS(t *testing.T) {
|
|||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
UseDNS64: true,
|
UseDNS64: true,
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
|
|||||||
@@ -103,16 +103,26 @@ type SystemResolvers interface {
|
|||||||
//
|
//
|
||||||
// The zero Server is empty and ready for use.
|
// The zero Server is empty and ready for use.
|
||||||
type Server struct {
|
type Server struct {
|
||||||
// dnsProxy is the DNS proxy for forwarding client's DNS requests.
|
// addrProc, if not nil, is used to process clients' IP addresses with rDNS,
|
||||||
dnsProxy *proxy.Proxy
|
// WHOIS, etc.
|
||||||
|
addrProc client.AddressProcessor
|
||||||
|
|
||||||
// dnsFilter is the DNS filter for filtering client's DNS requests and
|
// bootstrap is the resolver for upstreams' hostnames.
|
||||||
// responses.
|
bootstrap upstream.Resolver
|
||||||
dnsFilter *filtering.DNSFilter
|
|
||||||
|
// clientIDCache is a temporary storage for ClientIDs that were extracted
|
||||||
|
// during the BeforeRequestHandler stage.
|
||||||
|
clientIDCache cache.Cache
|
||||||
|
|
||||||
// dhcpServer is the DHCP server for accessing lease data.
|
// dhcpServer is the DHCP server for accessing lease data.
|
||||||
dhcpServer DHCP
|
dhcpServer DHCP
|
||||||
|
|
||||||
|
// etcHosts contains the current data from the system's hosts files.
|
||||||
|
etcHosts upstream.Resolver
|
||||||
|
|
||||||
|
// privateNets is the configured set of IP networks considered private.
|
||||||
|
privateNets netutil.SubnetSet
|
||||||
|
|
||||||
// queryLog is the query log for client's DNS requests, responses and
|
// queryLog is the query log for client's DNS requests, responses and
|
||||||
// filtering results.
|
// filtering results.
|
||||||
queryLog querylog.QueryLog
|
queryLog querylog.QueryLog
|
||||||
@@ -120,37 +130,43 @@ type Server struct {
|
|||||||
// stats is the statistics collector for client's DNS usage data.
|
// stats is the statistics collector for client's DNS usage data.
|
||||||
stats stats.Interface
|
stats stats.Interface
|
||||||
|
|
||||||
|
// sysResolvers used to fetch system resolvers to use by default for private
|
||||||
|
// PTR resolving.
|
||||||
|
sysResolvers SystemResolvers
|
||||||
|
|
||||||
// access drops disallowed clients.
|
// access drops disallowed clients.
|
||||||
access *accessManager
|
access *accessManager
|
||||||
|
|
||||||
|
// anonymizer masks the client's IP addresses if needed.
|
||||||
|
anonymizer *aghnet.IPMut
|
||||||
|
|
||||||
// baseLogger is used to create loggers for other entities. It should not
|
// baseLogger is used to create loggers for other entities. It should not
|
||||||
// have a prefix and must not be nil.
|
// have a prefix and must not be nil.
|
||||||
baseLogger *slog.Logger
|
baseLogger *slog.Logger
|
||||||
|
|
||||||
// localDomainSuffix is the suffix used to detect internal hosts. It
|
// dnsFilter is the DNS filter for filtering client's DNS requests and
|
||||||
// must be a valid domain name plus dots on each side.
|
// responses.
|
||||||
localDomainSuffix string
|
dnsFilter *filtering.DNSFilter
|
||||||
|
|
||||||
|
// dnsProxy is the DNS proxy for forwarding client's DNS requests.
|
||||||
|
dnsProxy *proxy.Proxy
|
||||||
|
|
||||||
|
// internalProxy resolves internal requests from the application itself. It
|
||||||
|
// isn't started and so no listen ports are required.
|
||||||
|
internalProxy *proxy.Proxy
|
||||||
|
|
||||||
// ipset processes DNS requests using ipset data. It must not be nil after
|
// ipset processes DNS requests using ipset data. It must not be nil after
|
||||||
// initialization. See [newIpsetHandler].
|
// initialization. See [newIpsetHandler].
|
||||||
ipset *ipsetHandler
|
ipset *ipsetHandler
|
||||||
|
|
||||||
// privateNets is the configured set of IP networks considered private.
|
// dns64Pref is the NAT64 prefix used for DNS64 response mapping. The major
|
||||||
privateNets netutil.SubnetSet
|
// part of DNS64 happens inside the [proxy] package, but there still are
|
||||||
|
// some places where response mapping is needed (e.g. DHCP).
|
||||||
|
dns64Pref netip.Prefix
|
||||||
|
|
||||||
// addrProc, if not nil, is used to process clients' IP addresses with rDNS,
|
// localDomainSuffix is the suffix used to detect internal hosts. It
|
||||||
// WHOIS, etc.
|
// must be a valid domain name plus dots on each side.
|
||||||
addrProc client.AddressProcessor
|
localDomainSuffix string
|
||||||
|
|
||||||
// sysResolvers used to fetch system resolvers to use by default for private
|
|
||||||
// PTR resolving.
|
|
||||||
sysResolvers SystemResolvers
|
|
||||||
|
|
||||||
// etcHosts contains the current data from the system's hosts files.
|
|
||||||
etcHosts upstream.Resolver
|
|
||||||
|
|
||||||
// bootstrap is the resolver for upstreams' hostnames.
|
|
||||||
bootstrap upstream.Resolver
|
|
||||||
|
|
||||||
// bootResolvers are the resolvers that should be used for
|
// bootResolvers are the resolvers that should be used for
|
||||||
// bootstrapping along with [etcHosts].
|
// bootstrapping along with [etcHosts].
|
||||||
@@ -159,34 +175,26 @@ type Server struct {
|
|||||||
// [upstream.Resolver] interface.
|
// [upstream.Resolver] interface.
|
||||||
bootResolvers []*upstream.UpstreamResolver
|
bootResolvers []*upstream.UpstreamResolver
|
||||||
|
|
||||||
// dns64Pref is the NAT64 prefix used for DNS64 response mapping. The major
|
// dnsNames are the DNS names from certificate (SAN) or CN value from
|
||||||
// part of DNS64 happens inside the [proxy] package, but there still are
|
// Subject.
|
||||||
// some places where response mapping is needed (e.g. DHCP).
|
dnsNames []string
|
||||||
dns64Pref netip.Prefix
|
|
||||||
|
|
||||||
// anonymizer masks the client's IP addresses if needed.
|
|
||||||
anonymizer *aghnet.IPMut
|
|
||||||
|
|
||||||
// clientIDCache is a temporary storage for ClientIDs that were extracted
|
|
||||||
// during the BeforeRequestHandler stage.
|
|
||||||
clientIDCache cache.Cache
|
|
||||||
|
|
||||||
// internalProxy resolves internal requests from the application itself. It
|
|
||||||
// isn't started and so no listen ports are required.
|
|
||||||
internalProxy *proxy.Proxy
|
|
||||||
|
|
||||||
// isRunning is true if the DNS server is running.
|
|
||||||
isRunning bool
|
|
||||||
|
|
||||||
// protectionUpdateInProgress is used to make sure that only one goroutine
|
|
||||||
// updating the protection configuration after a pause is running at a time.
|
|
||||||
protectionUpdateInProgress atomic.Bool
|
|
||||||
|
|
||||||
// conf is the current configuration of the server.
|
// conf is the current configuration of the server.
|
||||||
conf ServerConfig
|
conf ServerConfig
|
||||||
|
|
||||||
// serverLock protects Server.
|
// serverLock protects Server.
|
||||||
serverLock sync.RWMutex
|
serverLock sync.RWMutex
|
||||||
|
|
||||||
|
// protectionUpdateInProgress is used to make sure that only one goroutine
|
||||||
|
// updating the protection configuration after a pause is running at a time.
|
||||||
|
protectionUpdateInProgress atomic.Bool
|
||||||
|
|
||||||
|
// isRunning is true if the DNS server is running.
|
||||||
|
isRunning bool
|
||||||
|
|
||||||
|
// hasIPAddrs is set during the certificate parsing and is true if the
|
||||||
|
// configured certificate contains at least a single IP address.
|
||||||
|
hasIPAddrs bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// defaultLocalDomainSuffix is the default suffix used to detect internal hosts
|
// defaultLocalDomainSuffix is the default suffix used to detect internal hosts
|
||||||
|
|||||||
@@ -213,17 +213,23 @@ func createServerTLSConfig(t *testing.T) (*tls.Config, []byte, []byte) {
|
|||||||
}, certPem, keyPem
|
}, certPem, keyPem
|
||||||
}
|
}
|
||||||
|
|
||||||
func createTestTLS(t *testing.T, tlsConf TLSConfig) (s *Server, certPem []byte) {
|
func createTestTLS(t *testing.T, tlsConf *TLSConfig) (s *Server, certPem []byte) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
|
||||||
var keyPem []byte
|
var keyPem []byte
|
||||||
_, certPem, keyPem = createServerTLSConfig(t)
|
_, certPem, keyPem = createServerTLSConfig(t)
|
||||||
|
|
||||||
|
cert, err := tls.X509KeyPair(certPem, keyPem)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
tlsConf.Cert = &cert
|
||||||
|
|
||||||
s = createTestServer(t, &filtering.Config{
|
s = createTestServer(t, &filtering.Config{
|
||||||
BlockingMode: filtering.BlockingModeDefault,
|
BlockingMode: filtering.BlockingModeDefault,
|
||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: tlsConf,
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
@@ -232,10 +238,7 @@ func createTestTLS(t *testing.T, tlsConf TLSConfig) (s *Server, certPem []byte)
|
|||||||
ServePlainDNS: true,
|
ServePlainDNS: true,
|
||||||
})
|
})
|
||||||
|
|
||||||
tlsConf.CertificateChainData, tlsConf.PrivateKeyData = certPem, keyPem
|
err = s.Prepare(&s.conf)
|
||||||
s.conf.TLSConfig = tlsConf
|
|
||||||
|
|
||||||
err := s.Prepare(&s.conf)
|
|
||||||
require.NoErrorf(t, err, "failed to prepare server: %s", err)
|
require.NoErrorf(t, err, "failed to prepare server: %s", err)
|
||||||
|
|
||||||
return s, certPem
|
return s, certPem
|
||||||
@@ -354,6 +357,7 @@ func TestServer(t *testing.T) {
|
|||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
@@ -395,6 +399,7 @@ func TestServer_timeout(t *testing.T) {
|
|||||||
t.Run("custom", func(t *testing.T) {
|
t.Run("custom", func(t *testing.T) {
|
||||||
srvConf := &ServerConfig{
|
srvConf := &ServerConfig{
|
||||||
UpstreamTimeout: testTimeout,
|
UpstreamTimeout: testTimeout,
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
@@ -422,6 +427,7 @@ func TestServer_timeout(t *testing.T) {
|
|||||||
})
|
})
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
s.conf.TLSConf = &TLSConfig{}
|
||||||
s.conf.Config.UpstreamMode = UpstreamModeLoadBalance
|
s.conf.Config.UpstreamMode = UpstreamModeLoadBalance
|
||||||
s.conf.Config.EDNSClientSubnet = &EDNSClientSubnet{
|
s.conf.Config.EDNSClientSubnet = &EDNSClientSubnet{
|
||||||
Enabled: false,
|
Enabled: false,
|
||||||
@@ -436,6 +442,7 @@ func TestServer_timeout(t *testing.T) {
|
|||||||
|
|
||||||
func TestServer_Prepare_fallbacks(t *testing.T) {
|
func TestServer_Prepare_fallbacks(t *testing.T) {
|
||||||
srvConf := &ServerConfig{
|
srvConf := &ServerConfig{
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
FallbackDNS: []string{
|
FallbackDNS: []string{
|
||||||
"#tls://1.1.1.1",
|
"#tls://1.1.1.1",
|
||||||
@@ -466,6 +473,7 @@ func TestServerWithProtectionDisabled(t *testing.T) {
|
|||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
@@ -487,7 +495,7 @@ func TestServerWithProtectionDisabled(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestDoTServer(t *testing.T) {
|
func TestDoTServer(t *testing.T) {
|
||||||
s, certPem := createTestTLS(t, TLSConfig{
|
s, certPem := createTestTLS(t, &TLSConfig{
|
||||||
TLSListenAddrs: []*net.TCPAddr{{}},
|
TLSListenAddrs: []*net.TCPAddr{{}},
|
||||||
})
|
})
|
||||||
s.conf.UpstreamConfig.Upstreams = []upstream.Upstream{newGoogleUpstream()}
|
s.conf.UpstreamConfig.Upstreams = []upstream.Upstream{newGoogleUpstream()}
|
||||||
@@ -511,7 +519,7 @@ func TestDoTServer(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestDoQServer(t *testing.T) {
|
func TestDoQServer(t *testing.T) {
|
||||||
s, _ := createTestTLS(t, TLSConfig{
|
s, _ := createTestTLS(t, &TLSConfig{
|
||||||
QUICListenAddrs: []*net.UDPAddr{{IP: net.IP{127, 0, 0, 1}}},
|
QUICListenAddrs: []*net.UDPAddr{{IP: net.IP{127, 0, 0, 1}}},
|
||||||
})
|
})
|
||||||
s.conf.UpstreamConfig.Upstreams = []upstream.Upstream{newGoogleUpstream()}
|
s.conf.UpstreamConfig.Upstreams = []upstream.Upstream{newGoogleUpstream()}
|
||||||
@@ -596,6 +604,7 @@ func TestSafeSearch(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -690,6 +699,7 @@ func TestInvalidRequest(t *testing.T) {
|
|||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -721,6 +731,7 @@ func TestBlockedRequest(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -758,6 +769,7 @@ func TestServerCustomClientUpstream(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
CacheSize: defaultCacheSize,
|
CacheSize: defaultCacheSize,
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
@@ -838,6 +850,7 @@ func TestBlockCNAMEProtectionEnabled(t *testing.T) {
|
|||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -873,6 +886,7 @@ func TestBlockCNAME(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -947,6 +961,7 @@ func TestClientRulesForCNAMEMatching(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -994,6 +1009,7 @@ func TestNullBlockedRequest(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -1064,6 +1080,7 @@ func TestBlockedCustomIP(t *testing.T) {
|
|||||||
conf := &ServerConfig{
|
conf := &ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamDNS: []string{"8.8.8.8:53", "8.8.4.4:53"},
|
UpstreamDNS: []string{"8.8.8.8:53", "8.8.4.4:53"},
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
@@ -1119,6 +1136,7 @@ func TestBlockedByHosts(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -1172,6 +1190,7 @@ func TestBlockedBySafeBrowsing(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
@@ -1235,6 +1254,7 @@ func TestRewrite(t *testing.T) {
|
|||||||
assert.NoError(t, s.Prepare(&ServerConfig{
|
assert.NoError(t, s.Prepare(&ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamDNS: []string{"8.8.8.8:53"},
|
UpstreamDNS: []string{"8.8.8.8:53"},
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
@@ -1369,6 +1389,7 @@ func TestPTRResponseFromDHCPLeases(t *testing.T) {
|
|||||||
s.conf.UDPListenAddrs = []*net.UDPAddr{{}}
|
s.conf.UDPListenAddrs = []*net.UDPAddr{{}}
|
||||||
s.conf.TCPListenAddrs = []*net.TCPAddr{{}}
|
s.conf.TCPListenAddrs = []*net.TCPAddr{{}}
|
||||||
s.conf.UpstreamDNS = []string{"127.0.0.1:53"}
|
s.conf.UpstreamDNS = []string{"127.0.0.1:53"}
|
||||||
|
s.conf.TLSConf = &TLSConfig{}
|
||||||
s.conf.Config.EDNSClientSubnet = &EDNSClientSubnet{Enabled: false}
|
s.conf.Config.EDNSClientSubnet = &EDNSClientSubnet{Enabled: false}
|
||||||
s.conf.Config.ClientsContainer = EmptyClientsContainer{}
|
s.conf.Config.ClientsContainer = EmptyClientsContainer{}
|
||||||
s.conf.Config.UpstreamMode = UpstreamModeLoadBalance
|
s.conf.Config.UpstreamMode = UpstreamModeLoadBalance
|
||||||
@@ -1457,6 +1478,7 @@ func TestPTRResponseFromHosts(t *testing.T) {
|
|||||||
s.conf.UDPListenAddrs = []*net.UDPAddr{{}}
|
s.conf.UDPListenAddrs = []*net.UDPAddr{{}}
|
||||||
s.conf.TCPListenAddrs = []*net.TCPAddr{{}}
|
s.conf.TCPListenAddrs = []*net.TCPAddr{{}}
|
||||||
s.conf.UpstreamDNS = []string{"127.0.0.1:53"}
|
s.conf.UpstreamDNS = []string{"127.0.0.1:53"}
|
||||||
|
s.conf.TLSConf = &TLSConfig{}
|
||||||
s.conf.Config.EDNSClientSubnet = &EDNSClientSubnet{Enabled: false}
|
s.conf.Config.EDNSClientSubnet = &EDNSClientSubnet{Enabled: false}
|
||||||
s.conf.Config.ClientsContainer = EmptyClientsContainer{}
|
s.conf.Config.ClientsContainer = EmptyClientsContainer{}
|
||||||
s.conf.Config.UpstreamMode = UpstreamModeLoadBalance
|
s.conf.Config.UpstreamMode = UpstreamModeLoadBalance
|
||||||
@@ -1723,6 +1745,7 @@ func TestServer_Exchange(t *testing.T) {
|
|||||||
srv := createTestServer(t, &filtering.Config{
|
srv := createTestServer(t, &filtering.Config{
|
||||||
BlockingMode: filtering.BlockingModeDefault,
|
BlockingMode: filtering.BlockingModeDefault,
|
||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamDNS: []string{upsAddr},
|
UpstreamDNS: []string{upsAddr},
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
@@ -1746,6 +1769,7 @@ func TestServer_Exchange(t *testing.T) {
|
|||||||
srv := createTestServer(t, &filtering.Config{
|
srv := createTestServer(t, &filtering.Config{
|
||||||
BlockingMode: filtering.BlockingModeDefault,
|
BlockingMode: filtering.BlockingModeDefault,
|
||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamDNS: []string{upsAddr},
|
UpstreamDNS: []string{upsAddr},
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
|
|||||||
@@ -37,6 +37,7 @@ func TestServer_FilterDNSRewrite(t *testing.T) {
|
|||||||
srv := createTestServer(t, &filtering.Config{
|
srv := createTestServer(t, &filtering.Config{
|
||||||
BlockingMode: filtering.BlockingModeDefault,
|
BlockingMode: filtering.BlockingModeDefault,
|
||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ func TestHandleDNSRequest_handleDNSRequest(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{
|
EDNSClientSubnet: &EDNSClientSubnet{
|
||||||
|
|||||||
@@ -76,6 +76,7 @@ func TestDNSForwardHTTP_handleGetConfig(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{},
|
UDPListenAddrs: []*net.UDPAddr{},
|
||||||
TCPListenAddrs: []*net.TCPAddr{},
|
TCPListenAddrs: []*net.TCPAddr{},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamDNS: []string{"8.8.8.8:53", "8.8.4.4:53"},
|
UpstreamDNS: []string{"8.8.8.8:53", "8.8.4.4:53"},
|
||||||
FallbackDNS: []string{"9.9.9.10"},
|
FallbackDNS: []string{"9.9.9.10"},
|
||||||
@@ -159,6 +160,7 @@ func TestDNSForwardHTTP_handleSetConfig(t *testing.T) {
|
|||||||
forwardConf := ServerConfig{
|
forwardConf := ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{},
|
UDPListenAddrs: []*net.UDPAddr{},
|
||||||
TCPListenAddrs: []*net.TCPAddr{},
|
TCPListenAddrs: []*net.TCPAddr{},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamDNS: []string{"8.8.8.8:53", "8.8.4.4:53"},
|
UpstreamDNS: []string{"8.8.8.8:53", "8.8.4.4:53"},
|
||||||
RatelimitSubnetLenIPv4: 24,
|
RatelimitSubnetLenIPv4: 24,
|
||||||
@@ -369,6 +371,7 @@ func TestServer_HandleTestUpstreamDNS(t *testing.T) {
|
|||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
UpstreamTimeout: upsTimeout,
|
UpstreamTimeout: upsTimeout,
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
|
|||||||
@@ -246,9 +246,9 @@ func (s *Server) makeDDRResponse(req *dns.Msg) (resp *dns.Msg) {
|
|||||||
|
|
||||||
// TODO(e.burkov): Think about storing the FQDN version of the server's
|
// TODO(e.burkov): Think about storing the FQDN version of the server's
|
||||||
// name somewhere.
|
// name somewhere.
|
||||||
domainName := dns.Fqdn(s.conf.ServerName)
|
domainName := dns.Fqdn(s.conf.TLSConf.ServerName)
|
||||||
|
|
||||||
for _, addr := range s.conf.HTTPSListenAddrs {
|
for _, addr := range s.conf.TLSConf.HTTPSListenAddrs {
|
||||||
values := []dns.SVCBKeyValue{
|
values := []dns.SVCBKeyValue{
|
||||||
&dns.SVCBAlpn{Alpn: []string{"h2"}},
|
&dns.SVCBAlpn{Alpn: []string{"h2"}},
|
||||||
&dns.SVCBPort{Port: uint16(addr.Port)},
|
&dns.SVCBPort{Port: uint16(addr.Port)},
|
||||||
@@ -265,7 +265,7 @@ func (s *Server) makeDDRResponse(req *dns.Msg) (resp *dns.Msg) {
|
|||||||
resp.Answer = append(resp.Answer, ans)
|
resp.Answer = append(resp.Answer, ans)
|
||||||
}
|
}
|
||||||
|
|
||||||
if s.conf.hasIPAddrs {
|
if s.hasIPAddrs {
|
||||||
// Only add DNS-over-TLS resolvers in case the certificate contains IP
|
// Only add DNS-over-TLS resolvers in case the certificate contains IP
|
||||||
// addresses.
|
// addresses.
|
||||||
//
|
//
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ package dnsforward
|
|||||||
import (
|
import (
|
||||||
"cmp"
|
"cmp"
|
||||||
"context"
|
"context"
|
||||||
|
"crypto/tls"
|
||||||
"net"
|
"net"
|
||||||
"net/netip"
|
"net/netip"
|
||||||
"testing"
|
"testing"
|
||||||
@@ -77,6 +78,7 @@ func TestServer_ProcessInitial(t *testing.T) {
|
|||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
c := ServerConfig{
|
c := ServerConfig{
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
AAAADisabled: tc.aaaaDisabled,
|
AAAADisabled: tc.aaaaDisabled,
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
@@ -177,6 +179,7 @@ func TestServer_ProcessFilteringAfterResponse(t *testing.T) {
|
|||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
c := ServerConfig{
|
c := ServerConfig{
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
AAAADisabled: tc.aaaaDisabled,
|
AAAADisabled: tc.aaaaDisabled,
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
@@ -316,6 +319,8 @@ func TestServer_ProcessDDRQuery(t *testing.T) {
|
|||||||
}}
|
}}
|
||||||
|
|
||||||
_, certPem, keyPem := createServerTLSConfig(t)
|
_, certPem, keyPem := createServerTLSConfig(t)
|
||||||
|
cert, err := tls.X509KeyPair(certPem, keyPem)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
t.Run(tc.name, func(t *testing.T) {
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
@@ -328,19 +333,18 @@ func TestServer_ProcessDDRQuery(t *testing.T) {
|
|||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
ClientsContainer: EmptyClientsContainer{},
|
ClientsContainer: EmptyClientsContainer{},
|
||||||
},
|
},
|
||||||
TLSConfig: TLSConfig{
|
TLSConf: &TLSConfig{
|
||||||
ServerName: ddrTestDomainName,
|
ServerName: ddrTestDomainName,
|
||||||
CertificateChainData: certPem,
|
Cert: &cert,
|
||||||
PrivateKeyData: keyPem,
|
TLSListenAddrs: tc.addrsDoT,
|
||||||
TLSListenAddrs: tc.addrsDoT,
|
HTTPSListenAddrs: tc.addrsDoH,
|
||||||
HTTPSListenAddrs: tc.addrsDoH,
|
QUICListenAddrs: tc.addrsDoQ,
|
||||||
QUICListenAddrs: tc.addrsDoQ,
|
|
||||||
},
|
},
|
||||||
ServePlainDNS: true,
|
ServePlainDNS: true,
|
||||||
})
|
})
|
||||||
// TODO(e.burkov): Generate a certificate actually containing the
|
// TODO(e.burkov): Generate a certificate actually containing the
|
||||||
// IP addresses.
|
// IP addresses.
|
||||||
s.conf.hasIPAddrs = true
|
s.hasIPAddrs = true
|
||||||
|
|
||||||
req := createTestMessageWithType(tc.host, tc.qtype)
|
req := createTestMessageWithType(tc.host, tc.qtype)
|
||||||
|
|
||||||
@@ -657,6 +661,7 @@ func TestServer_HandleDNSRequest_restrictLocal(t *testing.T) {
|
|||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
// TODO(s.chzhen): Add tests where EDNSClientSubnet.Enabled is true.
|
// TODO(s.chzhen): Add tests where EDNSClientSubnet.Enabled is true.
|
||||||
// Improve Config declaration for tests.
|
// Improve Config declaration for tests.
|
||||||
Config: Config{
|
Config: Config{
|
||||||
@@ -789,6 +794,7 @@ func TestServer_ProcessUpstream_localPTR(t *testing.T) {
|
|||||||
ServerConfig{
|
ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
@@ -818,6 +824,7 @@ func TestServer_ProcessUpstream_localPTR(t *testing.T) {
|
|||||||
ServerConfig{
|
ServerConfig{
|
||||||
UDPListenAddrs: []*net.UDPAddr{{}},
|
UDPListenAddrs: []*net.UDPAddr{{}},
|
||||||
TCPListenAddrs: []*net.TCPAddr{{}},
|
TCPListenAddrs: []*net.TCPAddr{{}},
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ func TestGenAnswerHTTPS_andSVCB(t *testing.T) {
|
|||||||
s := createTestServer(t, &filtering.Config{
|
s := createTestServer(t, &filtering.Config{
|
||||||
BlockingMode: filtering.BlockingModeDefault,
|
BlockingMode: filtering.BlockingModeDefault,
|
||||||
}, ServerConfig{
|
}, ServerConfig{
|
||||||
|
TLSConf: &TLSConfig{},
|
||||||
Config: Config{
|
Config: Config{
|
||||||
UpstreamMode: UpstreamModeLoadBalance,
|
UpstreamMode: UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &EDNSClientSubnet{Enabled: false},
|
||||||
|
|||||||
@@ -1,19 +1,355 @@
|
|||||||
package home
|
package home
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
"net/netip"
|
"net/netip"
|
||||||
"net/textproto"
|
"net/textproto"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
|
||||||
"github.com/AdguardTeam/golibs/httphdr"
|
"github.com/AdguardTeam/golibs/httphdr"
|
||||||
|
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
||||||
"github.com/AdguardTeam/golibs/testutil"
|
"github.com/AdguardTeam/golibs/testutil"
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
"golang.org/x/crypto/bcrypt"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// TODO(s.chzhen): !! Add more tests.
|
||||||
|
func TestAuth_ServeHTTP_first_run(t *testing.T) {
|
||||||
|
storeGlobals(t)
|
||||||
|
|
||||||
|
globalContext.firstRun = true
|
||||||
|
|
||||||
|
mux := http.NewServeMux()
|
||||||
|
globalContext.mux = mux
|
||||||
|
|
||||||
|
var (
|
||||||
|
logger = slogutil.NewDiscardLogger()
|
||||||
|
ctx = testutil.ContextWithTimeout(t, testTimeout)
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
|
||||||
|
web, err := initWeb(ctx, options{}, nil, nil, logger, nil, false)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
globalContext.web = web
|
||||||
|
|
||||||
|
testCases := []struct {
|
||||||
|
url string
|
||||||
|
method string
|
||||||
|
code int
|
||||||
|
}{{
|
||||||
|
url: "/",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/apple/doh.mobileconfig",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/apple/dot.mobileconfig",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/i18n/change_language",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/i18n/current_language",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/install/check_config",
|
||||||
|
method: http.MethodPost,
|
||||||
|
code: http.StatusBadRequest,
|
||||||
|
}, {
|
||||||
|
url: "/control/install/configure",
|
||||||
|
method: http.MethodPost,
|
||||||
|
code: http.StatusBadRequest,
|
||||||
|
}, {
|
||||||
|
url: "/control/install/get_addresses",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusOK,
|
||||||
|
}, {
|
||||||
|
url: "/control/login",
|
||||||
|
method: http.MethodPost,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/logout",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/profile",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/profile/update",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/status",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/update",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/version.json",
|
||||||
|
method: http.MethodGet,
|
||||||
|
code: http.StatusFound,
|
||||||
|
}}
|
||||||
|
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.url, func(t *testing.T) {
|
||||||
|
r := httptest.NewRequest(tc.method, tc.url, nil)
|
||||||
|
|
||||||
|
h, pattern := mux.Handler(r)
|
||||||
|
require.NotEmpty(t, pattern)
|
||||||
|
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
h.ServeHTTP(w, r)
|
||||||
|
|
||||||
|
assert.Equal(t, tc.code, w.Code)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAuth_ServeHTTP(t *testing.T) {
|
||||||
|
storeGlobals(t)
|
||||||
|
|
||||||
|
const (
|
||||||
|
authNone = iota
|
||||||
|
authBasic
|
||||||
|
authCookie
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
testTTL = 60
|
||||||
|
userName = "name"
|
||||||
|
userPassword = "password"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
logger = slogutil.NewDiscardLogger()
|
||||||
|
ctx = testutil.ContextWithTimeout(t, testTimeout)
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
|
||||||
|
passwordHash, err := bcrypt.GenerateFromPassword([]byte(userPassword), bcrypt.DefaultCost)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
sessionsDB := filepath.Join(t.TempDir(), "sessions.db")
|
||||||
|
|
||||||
|
users := []webUser{{
|
||||||
|
Name: userName,
|
||||||
|
PasswordHash: string(passwordHash),
|
||||||
|
}}
|
||||||
|
auth := InitAuth(sessionsDB, users, testTTL, nil, nil)
|
||||||
|
globalContext.auth = auth
|
||||||
|
|
||||||
|
mux := http.NewServeMux()
|
||||||
|
globalContext.mux = mux
|
||||||
|
|
||||||
|
tlsMgr, err := newTLSManager(ctx, &tlsManagerConfig{
|
||||||
|
logger: logger,
|
||||||
|
configModified: func() {},
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
web, err := initWeb(ctx, options{}, nil, nil, logger, tlsMgr, false)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
globalContext.web = web
|
||||||
|
|
||||||
|
creds, err := json.Marshal(&loginJSON{Name: userName, Password: userPassword})
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
r := httptest.NewRequest(http.MethodPost, "/control/login", bytes.NewReader(creds))
|
||||||
|
r.Header.Set(httphdr.ContentType, aghhttp.HdrValApplicationJSON)
|
||||||
|
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
mux.ServeHTTP(w, r)
|
||||||
|
|
||||||
|
var loginCookie *http.Cookie
|
||||||
|
for _, c := range w.Result().Cookies() {
|
||||||
|
if c.Name == sessionCookieName {
|
||||||
|
loginCookie = c
|
||||||
|
}
|
||||||
|
}
|
||||||
|
require.NotNil(t, loginCookie)
|
||||||
|
|
||||||
|
testCases := []struct {
|
||||||
|
url string
|
||||||
|
method string
|
||||||
|
authMethod int
|
||||||
|
wantCode int
|
||||||
|
}{{
|
||||||
|
url: "/",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/i18n/change_language",
|
||||||
|
method: http.MethodPost,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusForbidden,
|
||||||
|
}, {
|
||||||
|
url: "/control/i18n/change_language",
|
||||||
|
method: http.MethodPost,
|
||||||
|
authMethod: authBasic,
|
||||||
|
wantCode: http.StatusInternalServerError,
|
||||||
|
}, {
|
||||||
|
url: "/control/i18n/change_language",
|
||||||
|
method: http.MethodPost,
|
||||||
|
authMethod: authCookie,
|
||||||
|
wantCode: http.StatusInternalServerError,
|
||||||
|
}, {
|
||||||
|
url: "/control/i18n/current_language",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusForbidden,
|
||||||
|
}, {
|
||||||
|
url: "/control/i18n/current_language",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authBasic,
|
||||||
|
wantCode: http.StatusOK,
|
||||||
|
}, {
|
||||||
|
url: "/control/i18n/current_language",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authCookie,
|
||||||
|
wantCode: http.StatusOK,
|
||||||
|
}, {
|
||||||
|
url: "/control/logout",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusForbidden,
|
||||||
|
}, {
|
||||||
|
url: "/control/logout",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authBasic,
|
||||||
|
wantCode: http.StatusFound,
|
||||||
|
}, {
|
||||||
|
url: "/control/profile",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusForbidden,
|
||||||
|
}, {
|
||||||
|
url: "/control/profile",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authBasic,
|
||||||
|
wantCode: http.StatusOK,
|
||||||
|
}, {
|
||||||
|
url: "/control/profile",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authCookie,
|
||||||
|
wantCode: http.StatusOK,
|
||||||
|
}, {
|
||||||
|
url: "/control/profile/update",
|
||||||
|
method: http.MethodPut,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusForbidden,
|
||||||
|
}, {
|
||||||
|
url: "/control/profile/update",
|
||||||
|
method: http.MethodPut,
|
||||||
|
authMethod: authBasic,
|
||||||
|
wantCode: http.StatusBadRequest,
|
||||||
|
}, {
|
||||||
|
url: "/control/profile/update",
|
||||||
|
method: http.MethodPut,
|
||||||
|
authMethod: authCookie,
|
||||||
|
wantCode: http.StatusBadRequest,
|
||||||
|
}, {
|
||||||
|
url: "/control/status",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusForbidden,
|
||||||
|
}, {
|
||||||
|
url: "/control/status",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authBasic,
|
||||||
|
wantCode: http.StatusOK,
|
||||||
|
}, {
|
||||||
|
url: "/control/status",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authCookie,
|
||||||
|
wantCode: http.StatusOK,
|
||||||
|
}, {
|
||||||
|
url: "/control/update",
|
||||||
|
method: http.MethodPost,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusForbidden,
|
||||||
|
}, {
|
||||||
|
url: "/control/version.json",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authNone,
|
||||||
|
wantCode: http.StatusForbidden,
|
||||||
|
}, {
|
||||||
|
url: "/control/version.json",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authBasic,
|
||||||
|
wantCode: http.StatusOK,
|
||||||
|
}, {
|
||||||
|
url: "/control/version.json",
|
||||||
|
method: http.MethodGet,
|
||||||
|
authMethod: authCookie,
|
||||||
|
wantCode: http.StatusOK,
|
||||||
|
}}
|
||||||
|
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.url, func(t *testing.T) {
|
||||||
|
r = httptest.NewRequest(tc.method, tc.url, nil)
|
||||||
|
switch tc.authMethod {
|
||||||
|
case authNone:
|
||||||
|
// Go on.
|
||||||
|
case authBasic:
|
||||||
|
r.SetBasicAuth(userName, userPassword)
|
||||||
|
case authCookie:
|
||||||
|
r.AddCookie(loginCookie)
|
||||||
|
default:
|
||||||
|
panic("unrecognized auth method")
|
||||||
|
}
|
||||||
|
|
||||||
|
h, pattern := mux.Handler(r)
|
||||||
|
require.NotEmpty(t, pattern)
|
||||||
|
|
||||||
|
w = httptest.NewRecorder()
|
||||||
|
h.ServeHTTP(w, r)
|
||||||
|
|
||||||
|
assert.Equal(t, tc.wantCode, w.Code)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Run("logout", func(t *testing.T) {
|
||||||
|
r = httptest.NewRequest(http.MethodGet, "/control/status", nil)
|
||||||
|
r.AddCookie(loginCookie)
|
||||||
|
w = httptest.NewRecorder()
|
||||||
|
|
||||||
|
mux.ServeHTTP(w, r)
|
||||||
|
assert.Equal(t, http.StatusOK, w.Code)
|
||||||
|
|
||||||
|
r = httptest.NewRequest(http.MethodGet, "/control/logout", nil)
|
||||||
|
r.AddCookie(loginCookie)
|
||||||
|
w = httptest.NewRecorder()
|
||||||
|
|
||||||
|
mux.ServeHTTP(w, r)
|
||||||
|
assert.Equal(t, http.StatusFound, w.Code)
|
||||||
|
|
||||||
|
r = httptest.NewRequest(http.MethodGet, "/control/status", nil)
|
||||||
|
r.AddCookie(loginCookie)
|
||||||
|
w = httptest.NewRecorder()
|
||||||
|
|
||||||
|
mux.ServeHTTP(w, r)
|
||||||
|
assert.Equal(t, http.StatusForbidden, w.Code)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
// implements http.ResponseWriter
|
// implements http.ResponseWriter
|
||||||
type testResponseWriter struct {
|
type testResponseWriter struct {
|
||||||
hdr http.Header
|
hdr http.Header
|
||||||
|
|||||||
@@ -28,6 +28,10 @@ type clientsContainer struct {
|
|||||||
// filter. It must not be nil.
|
// filter. It must not be nil.
|
||||||
baseLogger *slog.Logger
|
baseLogger *slog.Logger
|
||||||
|
|
||||||
|
// logger is used for logging the operation of the client container. It
|
||||||
|
// must not be nil.
|
||||||
|
logger *slog.Logger
|
||||||
|
|
||||||
// storage stores information about persistent clients.
|
// storage stores information about persistent clients.
|
||||||
storage *client.Storage
|
storage *client.Storage
|
||||||
|
|
||||||
@@ -58,6 +62,7 @@ type clientsContainer struct {
|
|||||||
// BlockedClientChecker checks if a client is blocked by the current access
|
// BlockedClientChecker checks if a client is blocked by the current access
|
||||||
// settings.
|
// settings.
|
||||||
type BlockedClientChecker interface {
|
type BlockedClientChecker interface {
|
||||||
|
// TODO(s.chzhen): Accept [client.FindParams].
|
||||||
IsBlockedClient(ip netip.Addr, clientID string) (blocked bool, rule string)
|
IsBlockedClient(ip netip.Addr, clientID string) (blocked bool, rule string)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -80,6 +85,7 @@ func (clients *clientsContainer) Init(
|
|||||||
}
|
}
|
||||||
|
|
||||||
clients.baseLogger = baseLogger
|
clients.baseLogger = baseLogger
|
||||||
|
clients.logger = baseLogger.With(slogutil.KeyPrefix, "client_container")
|
||||||
clients.safeSearchCacheSize = filteringConf.SafeSearchCacheSize
|
clients.safeSearchCacheSize = filteringConf.SafeSearchCacheSize
|
||||||
clients.safeSearchCacheTTL = time.Minute * time.Duration(filteringConf.CacheTime)
|
clients.safeSearchCacheTTL = time.Minute * time.Duration(filteringConf.CacheTime)
|
||||||
|
|
||||||
@@ -269,7 +275,7 @@ func (clients *clientsContainer) forConfig() (objs []*clientObject) {
|
|||||||
|
|
||||||
BlockedServices: cli.BlockedServices.Clone(),
|
BlockedServices: cli.BlockedServices.Clone(),
|
||||||
|
|
||||||
IDs: cli.IDs(),
|
IDs: cli.Identifiers(),
|
||||||
Tags: slices.Clone(cli.Tags),
|
Tags: slices.Clone(cli.Tags),
|
||||||
Upstreams: slices.Clone(cli.Upstreams),
|
Upstreams: slices.Clone(cli.Upstreams),
|
||||||
|
|
||||||
@@ -356,15 +362,27 @@ func (clients *clientsContainer) clientOrArtificial(
|
|||||||
}, true
|
}, true
|
||||||
}
|
}
|
||||||
|
|
||||||
// shouldCountClient is a wrapper around [clientsContainer.find] to make it a
|
// shouldCountClient is a wrapper around [client.Storage.Find] to make it a
|
||||||
// valid client information finder for the statistics. If no information about
|
// valid client information finder for the statistics. If no information about
|
||||||
// the client is found, it returns true.
|
// the client is found, it returns true. Values of ids must be either a valid
|
||||||
|
// ClientID or a valid IP address.
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Accept [client.FindParams].
|
||||||
func (clients *clientsContainer) shouldCountClient(ids []string) (y bool) {
|
func (clients *clientsContainer) shouldCountClient(ids []string) (y bool) {
|
||||||
clients.lock.Lock()
|
clients.lock.Lock()
|
||||||
defer clients.lock.Unlock()
|
defer clients.lock.Unlock()
|
||||||
|
|
||||||
|
params := &client.FindParams{}
|
||||||
for _, id := range ids {
|
for _, id := range ids {
|
||||||
client, ok := clients.storage.Find(id)
|
err := params.Set(id)
|
||||||
|
if err != nil {
|
||||||
|
// Should not happen.
|
||||||
|
clients.logger.Warn("parsing find params", slogutil.KeyError, err)
|
||||||
|
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
client, ok := clients.storage.Find(params)
|
||||||
if ok {
|
if ok {
|
||||||
return !client.IgnoreStatistics
|
return !client.IgnoreStatistics
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -300,7 +300,7 @@ func clientToJSON(c *client.Persistent) (cj *clientJSON) {
|
|||||||
|
|
||||||
return &clientJSON{
|
return &clientJSON{
|
||||||
Name: c.Name,
|
Name: c.Name,
|
||||||
IDs: c.IDs(),
|
IDs: c.Identifiers(),
|
||||||
Tags: c.Tags,
|
Tags: c.Tags,
|
||||||
UseGlobalSettings: !c.UseOwnSettings,
|
UseGlobalSettings: !c.UseOwnSettings,
|
||||||
FilteringEnabled: c.FilteringEnabled,
|
FilteringEnabled: c.FilteringEnabled,
|
||||||
@@ -428,32 +428,53 @@ func (clients *clientsContainer) handleUpdateClient(w http.ResponseWriter, r *ht
|
|||||||
// Deprecated: Remove it when migration to the new API is over.
|
// Deprecated: Remove it when migration to the new API is over.
|
||||||
func (clients *clientsContainer) handleFindClient(w http.ResponseWriter, r *http.Request) {
|
func (clients *clientsContainer) handleFindClient(w http.ResponseWriter, r *http.Request) {
|
||||||
q := r.URL.Query()
|
q := r.URL.Query()
|
||||||
data := []map[string]*clientJSON{}
|
data := make([]map[string]*clientJSON, 0, len(q))
|
||||||
|
params := &client.FindParams{}
|
||||||
|
var err error
|
||||||
|
|
||||||
for i := range len(q) {
|
for i := range len(q) {
|
||||||
idStr := q.Get(fmt.Sprintf("ip%d", i))
|
idStr := q.Get(fmt.Sprintf("ip%d", i))
|
||||||
if idStr == "" {
|
if idStr == "" {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
|
err = params.Set(idStr)
|
||||||
|
if err != nil {
|
||||||
|
clients.logger.DebugContext(
|
||||||
|
r.Context(),
|
||||||
|
"finding client",
|
||||||
|
"id", idStr,
|
||||||
|
slogutil.KeyError, err,
|
||||||
|
)
|
||||||
|
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
data = append(data, map[string]*clientJSON{
|
data = append(data, map[string]*clientJSON{
|
||||||
idStr: clients.findClient(idStr),
|
idStr: clients.findClient(idStr, params),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
aghhttp.WriteJSONResponseOK(w, r, data)
|
aghhttp.WriteJSONResponseOK(w, r, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
// findClient returns available information about a client by idStr from the
|
// findClient returns available information about a client by params from the
|
||||||
// client's storage or access settings. cj is guaranteed to be non-nil.
|
// client's storage or access settings. idStr is the string representation of
|
||||||
func (clients *clientsContainer) findClient(idStr string) (cj *clientJSON) {
|
// typed params. params must not be nil. cj is guaranteed to be non-nil.
|
||||||
ip, _ := netip.ParseAddr(idStr)
|
func (clients *clientsContainer) findClient(
|
||||||
c, ok := clients.storage.Find(idStr)
|
idStr string,
|
||||||
|
params *client.FindParams,
|
||||||
|
) (cj *clientJSON) {
|
||||||
|
c, ok := clients.storage.Find(params)
|
||||||
if !ok {
|
if !ok {
|
||||||
return clients.findRuntime(ip, idStr)
|
return clients.findRuntime(idStr, params)
|
||||||
}
|
}
|
||||||
|
|
||||||
cj = clientToJSON(c)
|
cj = clientToJSON(c)
|
||||||
disallowed, rule := clients.clientChecker.IsBlockedClient(ip, idStr)
|
disallowed, rule := clients.clientChecker.IsBlockedClient(
|
||||||
|
params.RemoteIP,
|
||||||
|
string(params.ClientID),
|
||||||
|
)
|
||||||
cj.Disallowed, cj.DisallowedRule = &disallowed, &rule
|
cj.Disallowed, cj.DisallowedRule = &disallowed, &rule
|
||||||
|
|
||||||
return cj
|
return cj
|
||||||
@@ -472,7 +493,8 @@ type searchClientJSON struct {
|
|||||||
ID string `json:"id"`
|
ID string `json:"id"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// handleSearchClient is the handler for the POST /control/clients/search HTTP API.
|
// handleSearchClient is the handler for the POST /control/clients/search HTTP
|
||||||
|
// API.
|
||||||
func (clients *clientsContainer) handleSearchClient(w http.ResponseWriter, r *http.Request) {
|
func (clients *clientsContainer) handleSearchClient(w http.ResponseWriter, r *http.Request) {
|
||||||
q := searchQueryJSON{}
|
q := searchQueryJSON{}
|
||||||
err := json.NewDecoder(r.Body).Decode(&q)
|
err := json.NewDecoder(r.Body).Decode(&q)
|
||||||
@@ -482,11 +504,25 @@ func (clients *clientsContainer) handleSearchClient(w http.ResponseWriter, r *ht
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
data := []map[string]*clientJSON{}
|
data := make([]map[string]*clientJSON, 0, len(q.Clients))
|
||||||
|
params := &client.FindParams{}
|
||||||
|
|
||||||
for _, c := range q.Clients {
|
for _, c := range q.Clients {
|
||||||
idStr := c.ID
|
idStr := c.ID
|
||||||
|
err = params.Set(idStr)
|
||||||
|
if err != nil {
|
||||||
|
clients.logger.DebugContext(
|
||||||
|
r.Context(),
|
||||||
|
"searching client",
|
||||||
|
"id", idStr,
|
||||||
|
slogutil.KeyError, err,
|
||||||
|
)
|
||||||
|
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
data = append(data, map[string]*clientJSON{
|
data = append(data, map[string]*clientJSON{
|
||||||
idStr: clients.findClient(idStr),
|
idStr: clients.findClient(idStr, params),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -494,38 +530,37 @@ func (clients *clientsContainer) handleSearchClient(w http.ResponseWriter, r *ht
|
|||||||
}
|
}
|
||||||
|
|
||||||
// findRuntime looks up the IP in runtime and temporary storages, like
|
// findRuntime looks up the IP in runtime and temporary storages, like
|
||||||
// /etc/hosts tables, DHCP leases, or blocklists. cj is guaranteed to be
|
// /etc/hosts tables, DHCP leases, or blocklists. params must not be nil. cj
|
||||||
// non-nil.
|
// is guaranteed to be non-nil.
|
||||||
func (clients *clientsContainer) findRuntime(ip netip.Addr, idStr string) (cj *clientJSON) {
|
func (clients *clientsContainer) findRuntime(
|
||||||
|
idStr string,
|
||||||
|
params *client.FindParams,
|
||||||
|
) (cj *clientJSON) {
|
||||||
|
var host string
|
||||||
|
whois := &whois.Info{}
|
||||||
|
|
||||||
|
ip := params.RemoteIP
|
||||||
rc := clients.storage.ClientRuntime(ip)
|
rc := clients.storage.ClientRuntime(ip)
|
||||||
if rc == nil {
|
if rc != nil {
|
||||||
// It is still possible that the IP used to be in the runtime clients
|
_, host = rc.Info()
|
||||||
// list, but then the server was reloaded. So, check the DNS server's
|
whois = whoisOrEmpty(rc)
|
||||||
// blocked IP list.
|
|
||||||
//
|
|
||||||
// See https://github.com/AdguardTeam/AdGuardHome/issues/2428.
|
|
||||||
disallowed, rule := clients.clientChecker.IsBlockedClient(ip, idStr)
|
|
||||||
cj = &clientJSON{
|
|
||||||
IDs: []string{idStr},
|
|
||||||
Disallowed: &disallowed,
|
|
||||||
DisallowedRule: &rule,
|
|
||||||
WHOIS: &whois.Info{},
|
|
||||||
}
|
|
||||||
|
|
||||||
return cj
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_, host := rc.Info()
|
// Check the DNS server's blocked IP list regardless of whether a runtime
|
||||||
cj = &clientJSON{
|
// client was found or not. This is because it's still possible that the
|
||||||
Name: host,
|
// runtime client associated with the IP address was stored previously, but
|
||||||
IDs: []string{idStr},
|
// then the server was reloaded.
|
||||||
WHOIS: whoisOrEmpty(rc),
|
//
|
||||||
|
// See https://github.com/AdguardTeam/AdGuardHome/issues/2428.
|
||||||
|
disallowed, rule := clients.clientChecker.IsBlockedClient(ip, string(params.ClientID))
|
||||||
|
|
||||||
|
return &clientJSON{
|
||||||
|
Name: host,
|
||||||
|
IDs: []string{idStr},
|
||||||
|
WHOIS: whois,
|
||||||
|
Disallowed: &disallowed,
|
||||||
|
DisallowedRule: &rule,
|
||||||
}
|
}
|
||||||
|
|
||||||
disallowed, rule := clients.clientChecker.IsBlockedClient(ip, idStr)
|
|
||||||
cj.Disallowed, cj.DisallowedRule = &disallowed, &rule
|
|
||||||
|
|
||||||
return cj
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// RegisterClientsHandlers registers HTTP handlers
|
// RegisterClientsHandlers registers HTTP handlers
|
||||||
|
|||||||
@@ -153,7 +153,7 @@ func TestClientsContainer_HandleAddClient(t *testing.T) {
|
|||||||
clientTwo := newPersistentClientWithIDs(t, "client2", []string{testClientIP2})
|
clientTwo := newPersistentClientWithIDs(t, "client2", []string{testClientIP2})
|
||||||
|
|
||||||
clientEmptyID := newPersistentClient("empty_client_id")
|
clientEmptyID := newPersistentClient("empty_client_id")
|
||||||
clientEmptyID.ClientIDs = []string{""}
|
clientEmptyID.ClientIDs = []client.ClientID{""}
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
name string
|
name string
|
||||||
@@ -278,7 +278,7 @@ func TestClientsContainer_HandleUpdateClient(t *testing.T) {
|
|||||||
clientModified := newPersistentClientWithIDs(t, "client2", []string{testClientIP2})
|
clientModified := newPersistentClientWithIDs(t, "client2", []string{testClientIP2})
|
||||||
|
|
||||||
clientEmptyID := newPersistentClient("empty_client_id")
|
clientEmptyID := newPersistentClient("empty_client_id")
|
||||||
clientEmptyID.ClientIDs = []string{""}
|
clientEmptyID.ClientIDs = []client.ClientID{""}
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
name string
|
name string
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import (
|
|||||||
"net/netip"
|
"net/netip"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"slices"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghalg"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghalg"
|
||||||
@@ -23,6 +24,7 @@ import (
|
|||||||
"github.com/AdguardTeam/golibs/log"
|
"github.com/AdguardTeam/golibs/log"
|
||||||
"github.com/AdguardTeam/golibs/netutil"
|
"github.com/AdguardTeam/golibs/netutil"
|
||||||
"github.com/AdguardTeam/golibs/timeutil"
|
"github.com/AdguardTeam/golibs/timeutil"
|
||||||
|
"github.com/google/go-cmp/cmp"
|
||||||
"github.com/google/renameio/v2/maybe"
|
"github.com/google/renameio/v2/maybe"
|
||||||
yaml "gopkg.in/yaml.v3"
|
yaml "gopkg.in/yaml.v3"
|
||||||
)
|
)
|
||||||
@@ -261,30 +263,128 @@ type dnsConfig struct {
|
|||||||
// HostsFileEnabled defines whether to use information from the system hosts
|
// HostsFileEnabled defines whether to use information from the system hosts
|
||||||
// file to resolve queries.
|
// file to resolve queries.
|
||||||
HostsFileEnabled bool `yaml:"hostsfile_enabled"`
|
HostsFileEnabled bool `yaml:"hostsfile_enabled"`
|
||||||
|
|
||||||
|
// PendingRequests configures duplicate requests policy.
|
||||||
|
PendingRequests *pendingRequests `yaml:"pending_requests"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type tlsConfigSettings struct {
|
// pendingRequests is a block with pending requests configuration.
|
||||||
Enabled bool `yaml:"enabled" json:"enabled"` // Enabled is the encryption (DoT/DoH/HTTPS) status
|
type pendingRequests struct {
|
||||||
ServerName string `yaml:"server_name" json:"server_name,omitempty"` // ServerName is the hostname of your HTTPS/TLS server
|
// Enabled controls if duplicate requests should be sent to the upstreams
|
||||||
ForceHTTPS bool `yaml:"force_https" json:"force_https"` // ForceHTTPS: if true, forces HTTP->HTTPS redirect
|
// along with the original one.
|
||||||
PortHTTPS uint16 `yaml:"port_https" json:"port_https,omitempty"` // HTTPS port. If 0, HTTPS will be disabled
|
Enabled bool `yaml:"enabled"`
|
||||||
PortDNSOverTLS uint16 `yaml:"port_dns_over_tls" json:"port_dns_over_tls,omitempty"` // DNS-over-TLS port. If 0, DoT will be disabled
|
}
|
||||||
PortDNSOverQUIC uint16 `yaml:"port_dns_over_quic" json:"port_dns_over_quic,omitempty"` // DNS-over-QUIC port. If 0, DoQ will be disabled
|
|
||||||
|
|
||||||
// PortDNSCrypt is the port for DNSCrypt requests. If it's zero,
|
// tlsConfigSettings is the TLS configuration for DNS-over-TLS, DNS-over-QUIC,
|
||||||
// DNSCrypt is disabled.
|
// and HTTPS. When adding new properties, update the [tlsConfigSettings.clone]
|
||||||
|
// and [tlsConfigSettings.setPrivateFieldsAndCompare] methods as necessary.
|
||||||
|
type tlsConfigSettings struct {
|
||||||
|
// Enabled indicates whether encryption (DoT/DoH/HTTPS) is enabled.
|
||||||
|
Enabled bool `yaml:"enabled" json:"enabled"`
|
||||||
|
|
||||||
|
// ServerName is the hostname of the HTTPS/TLS server.
|
||||||
|
ServerName string `yaml:"server_name" json:"server_name,omitempty"`
|
||||||
|
|
||||||
|
// ForceHTTPS, if true, forces an HTTP to HTTPS redirect.
|
||||||
|
ForceHTTPS bool `yaml:"force_https" json:"force_https"`
|
||||||
|
|
||||||
|
// PortHTTPS is the HTTPS port. If 0, HTTPS will be disabled.
|
||||||
|
PortHTTPS uint16 `yaml:"port_https" json:"port_https,omitempty"`
|
||||||
|
|
||||||
|
// PortDNSOverTLS is the DNS-over-TLS port. If 0, DoT will be disabled.
|
||||||
|
PortDNSOverTLS uint16 `yaml:"port_dns_over_tls" json:"port_dns_over_tls,omitempty"`
|
||||||
|
|
||||||
|
// PortDNSOverQUIC is the DNS-over-QUIC port. If 0, DoQ will be disabled.
|
||||||
|
PortDNSOverQUIC uint16 `yaml:"port_dns_over_quic" json:"port_dns_over_quic,omitempty"`
|
||||||
|
|
||||||
|
// PortDNSCrypt is the port for DNSCrypt requests. If it's zero, DNSCrypt
|
||||||
|
// is disabled.
|
||||||
PortDNSCrypt uint16 `yaml:"port_dnscrypt" json:"port_dnscrypt"`
|
PortDNSCrypt uint16 `yaml:"port_dnscrypt" json:"port_dnscrypt"`
|
||||||
// DNSCryptConfigFile is the path to the DNSCrypt config file. Must be
|
|
||||||
// set if PortDNSCrypt is not zero.
|
// DNSCryptConfigFile is the path to the DNSCrypt config file. Must be set
|
||||||
|
// if PortDNSCrypt is not zero.
|
||||||
//
|
//
|
||||||
// See https://github.com/AdguardTeam/dnsproxy and
|
// See https://github.com/AdguardTeam/dnsproxy and
|
||||||
// https://github.com/ameshkov/dnscrypt.
|
// https://github.com/ameshkov/dnscrypt.
|
||||||
DNSCryptConfigFile string `yaml:"dnscrypt_config_file" json:"dnscrypt_config_file"`
|
DNSCryptConfigFile string `yaml:"dnscrypt_config_file" json:"dnscrypt_config_file"`
|
||||||
|
|
||||||
// Allow DoH queries via unencrypted HTTP (e.g. for reverse proxying)
|
// AllowUnencryptedDoH allows DoH queries via unencrypted HTTP (e.g. for
|
||||||
|
// reverse proxying).
|
||||||
|
//
|
||||||
|
// TODO(s.chzhen): Add this option into the Web UI.
|
||||||
AllowUnencryptedDoH bool `yaml:"allow_unencrypted_doh" json:"allow_unencrypted_doh"`
|
AllowUnencryptedDoH bool `yaml:"allow_unencrypted_doh" json:"allow_unencrypted_doh"`
|
||||||
|
|
||||||
dnsforward.TLSConfig `yaml:",inline" json:",inline"`
|
// CertificateChain is the PEM-encoded certificate chain. Must be empty if
|
||||||
|
// [tlsConfigSettings.CertificatePath] is provided.
|
||||||
|
CertificateChain string `yaml:"certificate_chain" json:"certificate_chain"`
|
||||||
|
|
||||||
|
// PrivateKey is the PEM-encoded private key. Must be empty if
|
||||||
|
// [tlsConfigSettings.PrivateKeyPath] is provided.
|
||||||
|
PrivateKey string `yaml:"private_key" json:"private_key"`
|
||||||
|
|
||||||
|
// CertificatePath is the path to the certificate file. Must be empty if
|
||||||
|
// [tlsConfigSettings.CertificateChain] is provided.
|
||||||
|
CertificatePath string `yaml:"certificate_path" json:"certificate_path"`
|
||||||
|
|
||||||
|
// PrivateKeyPath is the path to the private key file. Must be empty if
|
||||||
|
// [tlsConfigSettings.PrivateKey] is provided.
|
||||||
|
PrivateKeyPath string `yaml:"private_key_path" json:"private_key_path"`
|
||||||
|
|
||||||
|
// OverrideTLSCiphers, when set, contains the names of the cipher suites to
|
||||||
|
// use. If the slice is empty, the default safe suites are used.
|
||||||
|
OverrideTLSCiphers []string `yaml:"override_tls_ciphers,omitempty" json:"-"`
|
||||||
|
|
||||||
|
// CertificateChainData is the PEM-encoded byte data for the certificate
|
||||||
|
// chain.
|
||||||
|
CertificateChainData []byte `yaml:"-" json:"-"`
|
||||||
|
|
||||||
|
// PrivateKeyData is the PEM-encoded byte data for the private key.
|
||||||
|
PrivateKeyData []byte `yaml:"-" json:"-"`
|
||||||
|
|
||||||
|
// StrictSNICheck controls if the connections with SNI mismatching the
|
||||||
|
// certificate's ones should be rejected.
|
||||||
|
StrictSNICheck bool `yaml:"strict_sni_check" json:"-"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// clone returns a deep copy of c.
|
||||||
|
func (c *tlsConfigSettings) clone() (clone *tlsConfigSettings) {
|
||||||
|
clone = &tlsConfigSettings{}
|
||||||
|
*clone = *c
|
||||||
|
|
||||||
|
clone.OverrideTLSCiphers = slices.Clone(c.OverrideTLSCiphers)
|
||||||
|
clone.CertificateChainData = slices.Clone(c.CertificateChainData)
|
||||||
|
clone.PrivateKeyData = slices.Clone(c.PrivateKeyData)
|
||||||
|
|
||||||
|
return clone
|
||||||
|
}
|
||||||
|
|
||||||
|
// setPrivateFieldsAndCompare sets any missing properties in conf to match those
|
||||||
|
// in c and returns true if TLS configurations are equal. conf must not be be
|
||||||
|
// nil.
|
||||||
|
// It sets the following properties because these are not accepted from the
|
||||||
|
// frontend:
|
||||||
|
//
|
||||||
|
// [tlsConfigSettings.AllowUnencryptedDoH]
|
||||||
|
// [tlsConfigSettings.DNSCryptConfigFile]
|
||||||
|
// [tlsConfigSettings.OverrideTLSCiphers]
|
||||||
|
// [tlsConfigSettings.PortDNSCrypt]
|
||||||
|
//
|
||||||
|
// The following properties are skipped as they are set by
|
||||||
|
// [tlsManager.loadTLSConfig]:
|
||||||
|
//
|
||||||
|
// [tlsConfigSettings.CertificateChainData]
|
||||||
|
// [tlsConfigSettings.PrivateKeyData]
|
||||||
|
func (c *tlsConfigSettings) setPrivateFieldsAndCompare(conf *tlsConfigSettings) (equal bool) {
|
||||||
|
conf.OverrideTLSCiphers = slices.Clone(c.OverrideTLSCiphers)
|
||||||
|
|
||||||
|
// TODO(s.chzhen): Remove this once the frontend supports it.
|
||||||
|
conf.AllowUnencryptedDoH = c.AllowUnencryptedDoH
|
||||||
|
|
||||||
|
conf.DNSCryptConfigFile = c.DNSCryptConfigFile
|
||||||
|
conf.PortDNSCrypt = c.PortDNSCrypt
|
||||||
|
|
||||||
|
// TODO(a.garipov): Define a custom comparer.
|
||||||
|
return cmp.Equal(c, conf)
|
||||||
}
|
}
|
||||||
|
|
||||||
type queryLogConfig struct {
|
type queryLogConfig struct {
|
||||||
@@ -380,6 +480,9 @@ var config = &configuration{
|
|||||||
UsePrivateRDNS: true,
|
UsePrivateRDNS: true,
|
||||||
ServePlainDNS: true,
|
ServePlainDNS: true,
|
||||||
HostsFileEnabled: true,
|
HostsFileEnabled: true,
|
||||||
|
PendingRequests: &pendingRequests{
|
||||||
|
Enabled: true,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
TLS: tlsConfigSettings{
|
TLS: tlsConfigSettings{
|
||||||
PortHTTPS: defaultPortHTTPS,
|
PortHTTPS: defaultPortHTTPS,
|
||||||
@@ -649,9 +752,8 @@ func (c *configuration) write(tlsMgr *tlsManager) (err error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if tlsMgr != nil {
|
if tlsMgr != nil {
|
||||||
tlsConf := tlsConfigSettings{}
|
tlsConf := tlsMgr.config()
|
||||||
tlsMgr.WriteDiskConfig(&tlsConf)
|
config.TLS = *tlsConf
|
||||||
config.TLS = tlsConf
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if globalContext.stats != nil {
|
if globalContext.stats != nil {
|
||||||
|
|||||||
@@ -164,11 +164,8 @@ func (vr *versionResponse) setAllowedToAutoUpdate(tlsMgr *tlsManager) (err error
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
tlsConf := &tlsConfigSettings{}
|
|
||||||
tlsMgr.WriteDiskConfig(tlsConf)
|
|
||||||
|
|
||||||
canUpdate := true
|
canUpdate := true
|
||||||
if tlsConfUsesPrivilegedPorts(tlsConf) ||
|
if tlsConfUsesPrivilegedPorts(tlsMgr.config()) ||
|
||||||
config.HTTPConfig.Address.Port() < 1024 ||
|
config.HTTPConfig.Address.Port() < 1024 ||
|
||||||
config.DNS.Port < 1024 {
|
config.DNS.Port < 1024 {
|
||||||
canUpdate, err = aghnet.CanBindPrivilegedPorts()
|
canUpdate, err = aghnet.CanBindPrivilegedPorts()
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package home
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"crypto/tls"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log/slog"
|
"log/slog"
|
||||||
"net"
|
"net"
|
||||||
@@ -111,9 +112,6 @@ func initDNS(
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
tlsConf := &tlsConfigSettings{}
|
|
||||||
tlsMgr.WriteDiskConfig(tlsConf)
|
|
||||||
|
|
||||||
return initDNSServer(
|
return initDNSServer(
|
||||||
globalContext.filters,
|
globalContext.filters,
|
||||||
globalContext.stats,
|
globalContext.stats,
|
||||||
@@ -121,7 +119,7 @@ func initDNS(
|
|||||||
globalContext.dhcpServer,
|
globalContext.dhcpServer,
|
||||||
anonymizer,
|
anonymizer,
|
||||||
httpRegister,
|
httpRegister,
|
||||||
tlsConf,
|
tlsMgr.config(),
|
||||||
tlsMgr,
|
tlsMgr,
|
||||||
baseLogger,
|
baseLogger,
|
||||||
)
|
)
|
||||||
@@ -255,11 +253,16 @@ func newServerConfig(
|
|||||||
fwdConf := dnsConf.Config
|
fwdConf := dnsConf.Config
|
||||||
fwdConf.ClientsContainer = clientsContainer
|
fwdConf.ClientsContainer = clientsContainer
|
||||||
|
|
||||||
|
intTLSConf, err := newDNSTLSConfig(tlsConf, hosts)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("constructing tls config: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
newConf = &dnsforward.ServerConfig{
|
newConf = &dnsforward.ServerConfig{
|
||||||
UDPListenAddrs: ipsToUDPAddrs(hosts, dnsConf.Port),
|
UDPListenAddrs: ipsToUDPAddrs(hosts, dnsConf.Port),
|
||||||
TCPListenAddrs: ipsToTCPAddrs(hosts, dnsConf.Port),
|
TCPListenAddrs: ipsToTCPAddrs(hosts, dnsConf.Port),
|
||||||
Config: fwdConf,
|
Config: fwdConf,
|
||||||
TLSConfig: newDNSTLSConfig(tlsConf, hosts),
|
TLSConf: intTLSConf,
|
||||||
TLSAllowUnencryptedDoH: tlsConf.AllowUnencryptedDoH,
|
TLSAllowUnencryptedDoH: tlsConf.AllowUnencryptedDoH,
|
||||||
UpstreamTimeout: time.Duration(dnsConf.UpstreamTimeout),
|
UpstreamTimeout: time.Duration(dnsConf.UpstreamTimeout),
|
||||||
TLSv12Roots: tlsMgr.rootCerts,
|
TLSv12Roots: tlsMgr.rootCerts,
|
||||||
@@ -272,6 +275,7 @@ func newServerConfig(
|
|||||||
ServeHTTP3: dnsConf.ServeHTTP3,
|
ServeHTTP3: dnsConf.ServeHTTP3,
|
||||||
UseHTTP3Upstreams: dnsConf.UseHTTP3Upstreams,
|
UseHTTP3Upstreams: dnsConf.UseHTTP3Upstreams,
|
||||||
ServePlainDNS: dnsConf.ServePlainDNS,
|
ServePlainDNS: dnsConf.ServePlainDNS,
|
||||||
|
PendingRequestsEnabled: dnsConf.PendingRequests.Enabled,
|
||||||
}
|
}
|
||||||
|
|
||||||
var initialAddresses []netip.Addr
|
var initialAddresses []netip.Addr
|
||||||
@@ -304,14 +308,19 @@ func newServerConfig(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// newDNSTLSConfig converts values from the configuration file into the internal
|
// newDNSTLSConfig converts values from the configuration file into the internal
|
||||||
// TLS settings for the DNS server. tlsConf must not be nil.
|
// TLS settings for the DNS server. conf must not be nil.
|
||||||
func newDNSTLSConfig(conf *tlsConfigSettings, addrs []netip.Addr) (dnsConf dnsforward.TLSConfig) {
|
func newDNSTLSConfig(
|
||||||
|
conf *tlsConfigSettings,
|
||||||
|
addrs []netip.Addr,
|
||||||
|
) (dnsConf *dnsforward.TLSConfig, err error) {
|
||||||
if !conf.Enabled {
|
if !conf.Enabled {
|
||||||
return dnsforward.TLSConfig{}
|
return &dnsforward.TLSConfig{}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
dnsConf = conf.TLSConfig
|
dnsConf = &dnsforward.TLSConfig{
|
||||||
dnsConf.ServerName = conf.ServerName
|
ServerName: conf.ServerName,
|
||||||
|
StrictSNICheck: conf.StrictSNICheck,
|
||||||
|
}
|
||||||
|
|
||||||
if conf.PortHTTPS != 0 {
|
if conf.PortHTTPS != 0 {
|
||||||
dnsConf.HTTPSListenAddrs = ipsToTCPAddrs(addrs, conf.PortHTTPS)
|
dnsConf.HTTPSListenAddrs = ipsToTCPAddrs(addrs, conf.PortHTTPS)
|
||||||
@@ -325,7 +334,22 @@ func newDNSTLSConfig(conf *tlsConfigSettings, addrs []netip.Addr) (dnsConf dnsfo
|
|||||||
dnsConf.QUICListenAddrs = ipsToUDPAddrs(addrs, conf.PortDNSOverQUIC)
|
dnsConf.QUICListenAddrs = ipsToUDPAddrs(addrs, conf.PortDNSOverQUIC)
|
||||||
}
|
}
|
||||||
|
|
||||||
return dnsConf
|
cert, err := tls.X509KeyPair(conf.CertificateChainData, conf.PrivateKeyData)
|
||||||
|
if err != nil {
|
||||||
|
const format = "parsing tls key pair: %w"
|
||||||
|
if conf.AllowUnencryptedDoH {
|
||||||
|
// TODO(s.chzhen): Use [slog.Logger].
|
||||||
|
log.Info("warning: %s: %s", format, err)
|
||||||
|
|
||||||
|
return dnsConf, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, fmt.Errorf(format, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
dnsConf.Cert = &cert
|
||||||
|
|
||||||
|
return dnsConf, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// newDNSCryptConfig converts values from the configuration file into the
|
// newDNSCryptConfig converts values from the configuration file into the
|
||||||
@@ -378,8 +402,7 @@ type dnsEncryption struct {
|
|||||||
// getDNSEncryption returns the TLS encryption addresses that AdGuard Home
|
// getDNSEncryption returns the TLS encryption addresses that AdGuard Home
|
||||||
// listens on. tlsMgr must not be nil.
|
// listens on. tlsMgr must not be nil.
|
||||||
func getDNSEncryption(tlsMgr *tlsManager) (de dnsEncryption) {
|
func getDNSEncryption(tlsMgr *tlsManager) (de dnsEncryption) {
|
||||||
tlsConf := tlsConfigSettings{}
|
tlsConf := tlsMgr.config()
|
||||||
tlsMgr.WriteDiskConfig(&tlsConf)
|
|
||||||
|
|
||||||
if !tlsConf.Enabled || len(tlsConf.ServerName) == 0 {
|
if !tlsConf.Enabled || len(tlsConf.ServerName) == 0 {
|
||||||
return dnsEncryption{}
|
return dnsEncryption{}
|
||||||
|
|||||||
@@ -991,9 +991,9 @@ func printWebAddrs(proto, addr string, port uint16) {
|
|||||||
//
|
//
|
||||||
// TODO(s.chzhen): Implement separate functions for HTTP and HTTPS.
|
// TODO(s.chzhen): Implement separate functions for HTTP and HTTPS.
|
||||||
func printHTTPAddresses(proto string, tlsMgr *tlsManager) {
|
func printHTTPAddresses(proto string, tlsMgr *tlsManager) {
|
||||||
tlsConf := tlsConfigSettings{}
|
var tlsConf *tlsConfigSettings
|
||||||
if tlsMgr != nil {
|
if tlsMgr != nil {
|
||||||
tlsMgr.WriteDiskConfig(&tlsConf)
|
tlsConf = tlsMgr.config()
|
||||||
}
|
}
|
||||||
|
|
||||||
port := config.HTTPConfig.Address.Port()
|
port := config.HTTPConfig.Address.Port()
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import (
|
|||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/dnsforward"
|
"github.com/AdguardTeam/AdGuardHome/internal/client"
|
||||||
"github.com/AdguardTeam/golibs/errors"
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
"github.com/AdguardTeam/golibs/httphdr"
|
"github.com/AdguardTeam/golibs/httphdr"
|
||||||
"github.com/AdguardTeam/golibs/log"
|
"github.com/AdguardTeam/golibs/log"
|
||||||
@@ -151,7 +151,7 @@ func handleMobileConfig(w http.ResponseWriter, r *http.Request, dnsp string) {
|
|||||||
|
|
||||||
clientID := q.Get("client_id")
|
clientID := q.Get("client_id")
|
||||||
if clientID != "" {
|
if clientID != "" {
|
||||||
err = dnsforward.ValidateClientID(clientID)
|
err = client.ValidateClientID(clientID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
respondJSONError(w, http.StatusBadRequest, err.Error())
|
respondJSONError(w, http.StatusBadRequest, err.Error())
|
||||||
|
|
||||||
|
|||||||
@@ -24,11 +24,9 @@ import (
|
|||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghhttp"
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghnet"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghnet"
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/aghtls"
|
"github.com/AdguardTeam/AdGuardHome/internal/aghtls"
|
||||||
"github.com/AdguardTeam/AdGuardHome/internal/dnsforward"
|
|
||||||
"github.com/AdguardTeam/golibs/errors"
|
"github.com/AdguardTeam/golibs/errors"
|
||||||
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
"github.com/AdguardTeam/golibs/logutil/slogutil"
|
||||||
"github.com/c2h5oh/datasize"
|
"github.com/c2h5oh/datasize"
|
||||||
"github.com/google/go-cmp/cmp"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// tlsManager contains the current configuration and state of AdGuard Home TLS
|
// tlsManager contains the current configuration and state of AdGuard Home TLS
|
||||||
@@ -37,6 +35,9 @@ type tlsManager struct {
|
|||||||
// logger is used for logging the operation of the TLS Manager.
|
// logger is used for logging the operation of the TLS Manager.
|
||||||
logger *slog.Logger
|
logger *slog.Logger
|
||||||
|
|
||||||
|
// mu protects status, certLastMod, conf, and servePlainDNS.
|
||||||
|
mu *sync.Mutex
|
||||||
|
|
||||||
// status is the current status of the configuration. It is never nil.
|
// status is the current status of the configuration. It is never nil.
|
||||||
status *tlsConfigStatus
|
status *tlsConfigStatus
|
||||||
|
|
||||||
@@ -52,6 +53,9 @@ type tlsManager struct {
|
|||||||
// Resolve it.
|
// Resolve it.
|
||||||
web *webAPI
|
web *webAPI
|
||||||
|
|
||||||
|
// conf contains the TLS configuration settings. It must not be nil.
|
||||||
|
conf *tlsConfigSettings
|
||||||
|
|
||||||
// configModified is called when the TLS configuration is changed via an
|
// configModified is called when the TLS configuration is changed via an
|
||||||
// HTTP request.
|
// HTTP request.
|
||||||
configModified func()
|
configModified func()
|
||||||
@@ -59,9 +63,6 @@ type tlsManager struct {
|
|||||||
// customCipherIDs are the ID of the cipher suites that AdGuard Home must use.
|
// customCipherIDs are the ID of the cipher suites that AdGuard Home must use.
|
||||||
customCipherIDs []uint16
|
customCipherIDs []uint16
|
||||||
|
|
||||||
confLock sync.Mutex
|
|
||||||
conf tlsConfigSettings
|
|
||||||
|
|
||||||
// servePlainDNS defines if plain DNS is allowed for incoming requests.
|
// servePlainDNS defines if plain DNS is allowed for incoming requests.
|
||||||
servePlainDNS bool
|
servePlainDNS bool
|
||||||
}
|
}
|
||||||
@@ -91,9 +92,10 @@ type tlsManagerConfig struct {
|
|||||||
func newTLSManager(ctx context.Context, conf *tlsManagerConfig) (m *tlsManager, err error) {
|
func newTLSManager(ctx context.Context, conf *tlsManagerConfig) (m *tlsManager, err error) {
|
||||||
m = &tlsManager{
|
m = &tlsManager{
|
||||||
logger: conf.logger,
|
logger: conf.logger,
|
||||||
|
mu: &sync.Mutex{},
|
||||||
configModified: conf.configModified,
|
configModified: conf.configModified,
|
||||||
status: &tlsConfigStatus{},
|
status: &tlsConfigStatus{},
|
||||||
conf: conf.tlsSettings,
|
conf: &conf.tlsSettings,
|
||||||
servePlainDNS: conf.servePlainDNS,
|
servePlainDNS: conf.servePlainDNS,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -112,17 +114,22 @@ func newTLSManager(ctx context.Context, conf *tlsManagerConfig) (m *tlsManager,
|
|||||||
m.logger.InfoContext(ctx, "using default ciphers")
|
m.logger.InfoContext(ctx, "using default ciphers")
|
||||||
}
|
}
|
||||||
|
|
||||||
if m.conf.Enabled {
|
m.mu.Lock()
|
||||||
err = m.load(ctx)
|
defer m.mu.Unlock()
|
||||||
if err != nil {
|
|
||||||
m.conf.Enabled = false
|
|
||||||
|
|
||||||
return m, err
|
if !m.conf.Enabled {
|
||||||
}
|
return m, nil
|
||||||
|
|
||||||
m.setCertFileTime(ctx)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
err = m.load(ctx)
|
||||||
|
if err != nil {
|
||||||
|
m.conf.Enabled = false
|
||||||
|
|
||||||
|
return m, err
|
||||||
|
}
|
||||||
|
|
||||||
|
m.setCertFileTime(ctx)
|
||||||
|
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -136,8 +143,9 @@ func (m *tlsManager) setWebAPI(webAPI *webAPI) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// load reloads the TLS configuration from files or data from the config file.
|
// load reloads the TLS configuration from files or data from the config file.
|
||||||
|
// m.mu is expected to be locked.
|
||||||
func (m *tlsManager) load(ctx context.Context) (err error) {
|
func (m *tlsManager) load(ctx context.Context) (err error) {
|
||||||
err = m.loadTLSConf(ctx, &m.conf, m.status)
|
err = m.loadTLSConfig(ctx, m.conf, m.status)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("loading config: %w", err)
|
return fmt.Errorf("loading config: %w", err)
|
||||||
}
|
}
|
||||||
@@ -145,15 +153,16 @@ func (m *tlsManager) load(ctx context.Context) (err error) {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// WriteDiskConfig - write config
|
// config returns a deep copy of the stored TLS configuration.
|
||||||
func (m *tlsManager) WriteDiskConfig(conf *tlsConfigSettings) {
|
func (m *tlsManager) config() (conf *tlsConfigSettings) {
|
||||||
m.confLock.Lock()
|
m.mu.Lock()
|
||||||
*conf = m.conf
|
defer m.mu.Unlock()
|
||||||
m.confLock.Unlock()
|
|
||||||
|
return m.conf.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
// setCertFileTime sets [tlsManager.certLastMod] from the certificate. If there
|
// setCertFileTime sets [tlsManager.certLastMod] from the certificate. If there
|
||||||
// are errors, setCertFileTime logs them.
|
// are errors, setCertFileTime logs them. m.mu is expected to be locked.
|
||||||
func (m *tlsManager) setCertFileTime(ctx context.Context) {
|
func (m *tlsManager) setCertFileTime(ctx context.Context) {
|
||||||
if len(m.conf.CertificatePath) == 0 {
|
if len(m.conf.CertificatePath) == 0 {
|
||||||
return
|
return
|
||||||
@@ -175,21 +184,21 @@ func (m *tlsManager) setCertFileTime(ctx context.Context) {
|
|||||||
func (m *tlsManager) start(_ context.Context) {
|
func (m *tlsManager) start(_ context.Context) {
|
||||||
m.registerWebHandlers()
|
m.registerWebHandlers()
|
||||||
|
|
||||||
m.confLock.Lock()
|
m.mu.Lock()
|
||||||
tlsConf := m.conf
|
defer m.mu.Unlock()
|
||||||
m.confLock.Unlock()
|
|
||||||
|
|
||||||
// The background context is used because the TLSConfigChanged wraps context
|
// The background context is used because the TLSConfigChanged wraps context
|
||||||
// with timeout on its own and shuts down the server, which handles current
|
// with timeout on its own and shuts down the server, which handles current
|
||||||
// request.
|
// request.
|
||||||
m.web.tlsConfigChanged(context.Background(), tlsConf)
|
m.web.tlsConfigChanged(context.Background(), m.conf)
|
||||||
}
|
}
|
||||||
|
|
||||||
// reload updates the configuration and restarts the TLS manager.
|
// reload updates the configuration and restarts the TLS manager.
|
||||||
func (m *tlsManager) reload(ctx context.Context) {
|
func (m *tlsManager) reload(ctx context.Context) {
|
||||||
m.confLock.Lock()
|
m.mu.Lock()
|
||||||
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
tlsConf := m.conf
|
tlsConf := m.conf
|
||||||
m.confLock.Unlock()
|
|
||||||
|
|
||||||
if !tlsConf.Enabled || len(tlsConf.CertificatePath) == 0 {
|
if !tlsConf.Enabled || len(tlsConf.CertificatePath) == 0 {
|
||||||
return
|
return
|
||||||
@@ -211,9 +220,7 @@ func (m *tlsManager) reload(ctx context.Context) {
|
|||||||
|
|
||||||
m.logger.InfoContext(ctx, "certificate file is modified")
|
m.logger.InfoContext(ctx, "certificate file is modified")
|
||||||
|
|
||||||
m.confLock.Lock()
|
|
||||||
err = m.load(ctx)
|
err = m.load(ctx)
|
||||||
m.confLock.Unlock()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
m.logger.ErrorContext(ctx, "reloading", slogutil.KeyError, err)
|
m.logger.ErrorContext(ctx, "reloading", slogutil.KeyError, err)
|
||||||
|
|
||||||
@@ -227,10 +234,6 @@ func (m *tlsManager) reload(ctx context.Context) {
|
|||||||
m.logger.ErrorContext(ctx, "reconfiguring dns server", slogutil.KeyError, err)
|
m.logger.ErrorContext(ctx, "reconfiguring dns server", slogutil.KeyError, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.confLock.Lock()
|
|
||||||
tlsConf = m.conf
|
|
||||||
m.confLock.Unlock()
|
|
||||||
|
|
||||||
// The background context is used because the TLSConfigChanged wraps context
|
// The background context is used because the TLSConfigChanged wraps context
|
||||||
// with timeout on its own and shuts down the server, which handles current
|
// with timeout on its own and shuts down the server, which handles current
|
||||||
// request.
|
// request.
|
||||||
@@ -238,15 +241,12 @@ func (m *tlsManager) reload(ctx context.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// reconfigureDNSServer updates the DNS server configuration using the stored
|
// reconfigureDNSServer updates the DNS server configuration using the stored
|
||||||
// TLS settings.
|
// TLS settings. m.mu is expected to be locked.
|
||||||
func (m *tlsManager) reconfigureDNSServer() (err error) {
|
func (m *tlsManager) reconfigureDNSServer() (err error) {
|
||||||
tlsConf := &tlsConfigSettings{}
|
|
||||||
m.WriteDiskConfig(tlsConf)
|
|
||||||
|
|
||||||
newConf, err := newServerConfig(
|
newConf, err := newServerConfig(
|
||||||
&config.DNS,
|
&config.DNS,
|
||||||
config.Clients.Sources,
|
config.Clients.Sources,
|
||||||
tlsConf,
|
m.conf,
|
||||||
m,
|
m,
|
||||||
httpRegister,
|
httpRegister,
|
||||||
globalContext.clients.storage,
|
globalContext.clients.storage,
|
||||||
@@ -263,9 +263,11 @@ func (m *tlsManager) reconfigureDNSServer() (err error) {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// loadTLSConf loads and validates the TLS configuration. The returned error is
|
// loadTLSConfig loads and validates the TLS configuration. It also sets
|
||||||
// also set in status.WarningValidation.
|
// [tlsConfigSettings.CertificateChainData] and
|
||||||
func (m *tlsManager) loadTLSConf(
|
// [tlsConfigSettings.PrivateKeyData] properties. The returned error is also
|
||||||
|
// set in status.WarningValidation.
|
||||||
|
func (m *tlsManager) loadTLSConfig(
|
||||||
ctx context.Context,
|
ctx context.Context,
|
||||||
tlsConf *tlsConfigSettings,
|
tlsConf *tlsConfigSettings,
|
||||||
status *tlsConfigStatus,
|
status *tlsConfigStatus,
|
||||||
@@ -357,10 +359,10 @@ type tlsConfigStatus struct {
|
|||||||
KeyType string `json:"key_type,omitempty"`
|
KeyType string `json:"key_type,omitempty"`
|
||||||
|
|
||||||
// NotBefore is the NotBefore field of the first certificate in the chain.
|
// NotBefore is the NotBefore field of the first certificate in the chain.
|
||||||
NotBefore time.Time `json:"not_before,omitempty"`
|
NotBefore time.Time `json:"not_before"`
|
||||||
|
|
||||||
// NotAfter is the NotAfter field of the first certificate in the chain.
|
// NotAfter is the NotAfter field of the first certificate in the chain.
|
||||||
NotAfter time.Time `json:"not_after,omitempty"`
|
NotAfter time.Time `json:"not_after"`
|
||||||
|
|
||||||
// WarningValidation is a validation warning message with the issue
|
// WarningValidation is a validation warning message with the issue
|
||||||
// description.
|
// description.
|
||||||
@@ -410,15 +412,23 @@ type tlsConfigSettingsExt struct {
|
|||||||
|
|
||||||
// handleTLSStatus is the handler for the GET /control/tls/status HTTP API.
|
// handleTLSStatus is the handler for the GET /control/tls/status HTTP API.
|
||||||
func (m *tlsManager) handleTLSStatus(w http.ResponseWriter, r *http.Request) {
|
func (m *tlsManager) handleTLSStatus(w http.ResponseWriter, r *http.Request) {
|
||||||
m.confLock.Lock()
|
var tlsConf *tlsConfigSettings
|
||||||
|
var servePlainDNS bool
|
||||||
|
func() {
|
||||||
|
m.mu.Lock()
|
||||||
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
|
tlsConf = m.conf.clone()
|
||||||
|
servePlainDNS = m.servePlainDNS
|
||||||
|
}()
|
||||||
|
|
||||||
data := tlsConfig{
|
data := tlsConfig{
|
||||||
tlsConfigSettingsExt: tlsConfigSettingsExt{
|
tlsConfigSettingsExt: tlsConfigSettingsExt{
|
||||||
tlsConfigSettings: m.conf,
|
tlsConfigSettings: *tlsConf,
|
||||||
ServePlainDNS: aghalg.BoolToNullBool(m.servePlainDNS),
|
ServePlainDNS: aghalg.BoolToNullBool(servePlainDNS),
|
||||||
},
|
},
|
||||||
tlsConfigStatus: m.status,
|
tlsConfigStatus: m.status,
|
||||||
}
|
}
|
||||||
m.confLock.Unlock()
|
|
||||||
|
|
||||||
marshalTLS(w, r, data)
|
marshalTLS(w, r, data)
|
||||||
}
|
}
|
||||||
@@ -434,6 +444,9 @@ func (m *tlsManager) handleTLSValidate(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
m.mu.Lock()
|
||||||
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
if setts.PrivateKeySaved {
|
if setts.PrivateKeySaved {
|
||||||
setts.PrivateKey = m.conf.PrivateKey
|
setts.PrivateKey = m.conf.PrivateKey
|
||||||
}
|
}
|
||||||
@@ -449,7 +462,7 @@ func (m *tlsManager) handleTLSValidate(w http.ResponseWriter, r *http.Request) {
|
|||||||
// Skip the error check, since we are only interested in the value of
|
// Skip the error check, since we are only interested in the value of
|
||||||
// status.WarningValidation.
|
// status.WarningValidation.
|
||||||
status := &tlsConfigStatus{}
|
status := &tlsConfigStatus{}
|
||||||
_ = m.loadTLSConf(ctx, &setts.tlsConfigSettings, status)
|
_ = m.loadTLSConfig(ctx, &setts.tlsConfigSettings, status)
|
||||||
resp := tlsConfig{
|
resp := tlsConfig{
|
||||||
tlsConfigSettingsExt: setts,
|
tlsConfigSettingsExt: setts,
|
||||||
tlsConfigStatus: status,
|
tlsConfigStatus: status,
|
||||||
@@ -458,42 +471,23 @@ func (m *tlsManager) handleTLSValidate(w http.ResponseWriter, r *http.Request) {
|
|||||||
marshalTLS(w, r, resp)
|
marshalTLS(w, r, resp)
|
||||||
}
|
}
|
||||||
|
|
||||||
// setConfig updates manager conf with the given one.
|
// setConfig updates manager TLS configuration with the given one. m.mu is
|
||||||
|
// expected to be locked.
|
||||||
func (m *tlsManager) setConfig(
|
func (m *tlsManager) setConfig(
|
||||||
ctx context.Context,
|
ctx context.Context,
|
||||||
newConf tlsConfigSettings,
|
newConf tlsConfigSettings,
|
||||||
status *tlsConfigStatus,
|
status *tlsConfigStatus,
|
||||||
servePlain aghalg.NullBool,
|
servePlain aghalg.NullBool,
|
||||||
) (restartHTTPS bool) {
|
) (restartHTTPS bool) {
|
||||||
m.confLock.Lock()
|
if !m.conf.setPrivateFieldsAndCompare(&newConf) {
|
||||||
defer m.confLock.Unlock()
|
|
||||||
|
|
||||||
// Reset the DNSCrypt data before comparing, since we currently do not
|
|
||||||
// accept these from the frontend.
|
|
||||||
//
|
|
||||||
// TODO(a.garipov): Define a custom comparer for dnsforward.TLSConfig.
|
|
||||||
newConf.DNSCryptConfigFile = m.conf.DNSCryptConfigFile
|
|
||||||
newConf.PortDNSCrypt = m.conf.PortDNSCrypt
|
|
||||||
if !cmp.Equal(m.conf, newConf, cmp.AllowUnexported(dnsforward.TLSConfig{})) {
|
|
||||||
m.logger.InfoContext(ctx, "config has changed, restarting https server")
|
m.logger.InfoContext(ctx, "config has changed, restarting https server")
|
||||||
restartHTTPS = true
|
restartHTTPS = true
|
||||||
} else {
|
} else {
|
||||||
m.logger.InfoContext(ctx, "config has not changed")
|
m.logger.InfoContext(ctx, "config has not changed")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note: don't do just `t.conf = data` because we must preserve all other members of t.conf
|
m.conf = &newConf
|
||||||
m.conf.Enabled = newConf.Enabled
|
|
||||||
m.conf.ServerName = newConf.ServerName
|
|
||||||
m.conf.ForceHTTPS = newConf.ForceHTTPS
|
|
||||||
m.conf.PortHTTPS = newConf.PortHTTPS
|
|
||||||
m.conf.PortDNSOverTLS = newConf.PortDNSOverTLS
|
|
||||||
m.conf.PortDNSOverQUIC = newConf.PortDNSOverQUIC
|
|
||||||
m.conf.CertificateChain = newConf.CertificateChain
|
|
||||||
m.conf.CertificatePath = newConf.CertificatePath
|
|
||||||
m.conf.CertificateChainData = newConf.CertificateChainData
|
|
||||||
m.conf.PrivateKey = newConf.PrivateKey
|
|
||||||
m.conf.PrivateKeyPath = newConf.PrivateKeyPath
|
|
||||||
m.conf.PrivateKeyData = newConf.PrivateKeyData
|
|
||||||
m.status = status
|
m.status = status
|
||||||
|
|
||||||
if servePlain != aghalg.NBNull {
|
if servePlain != aghalg.NBNull {
|
||||||
@@ -515,6 +509,16 @@ func (m *tlsManager) handleTLSConfigure(w http.ResponseWriter, r *http.Request)
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var restartHTTPS bool
|
||||||
|
defer func() {
|
||||||
|
if restartHTTPS {
|
||||||
|
m.configModified()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
m.mu.Lock()
|
||||||
|
defer m.mu.Unlock()
|
||||||
|
|
||||||
if req.PrivateKeySaved {
|
if req.PrivateKeySaved {
|
||||||
req.PrivateKey = m.conf.PrivateKey
|
req.PrivateKey = m.conf.PrivateKey
|
||||||
}
|
}
|
||||||
@@ -526,7 +530,7 @@ func (m *tlsManager) handleTLSConfigure(w http.ResponseWriter, r *http.Request)
|
|||||||
}
|
}
|
||||||
|
|
||||||
status := &tlsConfigStatus{}
|
status := &tlsConfigStatus{}
|
||||||
err = m.loadTLSConf(ctx, &req.tlsConfigSettings, status)
|
err = m.loadTLSConfig(ctx, &req.tlsConfigSettings, status)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
resp := tlsConfig{
|
resp := tlsConfig{
|
||||||
tlsConfigSettingsExt: req,
|
tlsConfigSettingsExt: req,
|
||||||
@@ -538,20 +542,18 @@ func (m *tlsManager) handleTLSConfigure(w http.ResponseWriter, r *http.Request)
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
restartHTTPS := m.setConfig(ctx, req.tlsConfigSettings, status, req.ServePlainDNS)
|
restartHTTPS = m.setConfig(ctx, req.tlsConfigSettings, status, req.ServePlainDNS)
|
||||||
m.setCertFileTime(ctx)
|
m.setCertFileTime(ctx)
|
||||||
|
|
||||||
if req.ServePlainDNS != aghalg.NBNull {
|
if req.ServePlainDNS != aghalg.NBNull {
|
||||||
func() {
|
func() {
|
||||||
m.confLock.Lock()
|
config.Lock()
|
||||||
defer m.confLock.Unlock()
|
defer config.Unlock()
|
||||||
|
|
||||||
config.DNS.ServePlainDNS = req.ServePlainDNS == aghalg.NBTrue
|
config.DNS.ServePlainDNS = req.ServePlainDNS == aghalg.NBTrue
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
m.configModified()
|
|
||||||
|
|
||||||
err = m.reconfigureDNSServer()
|
err = m.reconfigureDNSServer()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
m.logger.ErrorContext(ctx, "reconfiguring dns server", slogutil.KeyError, err)
|
m.logger.ErrorContext(ctx, "reconfiguring dns server", slogutil.KeyError, err)
|
||||||
@@ -567,18 +569,18 @@ func (m *tlsManager) handleTLSConfigure(w http.ResponseWriter, r *http.Request)
|
|||||||
}
|
}
|
||||||
|
|
||||||
marshalTLS(w, r, resp)
|
marshalTLS(w, r, resp)
|
||||||
if f, ok := w.(http.Flusher); ok {
|
rc := http.NewResponseController(w)
|
||||||
f.Flush()
|
err = rc.Flush()
|
||||||
|
if err != nil {
|
||||||
|
m.logger.ErrorContext(ctx, "flushing response", slogutil.KeyError, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// The background context is used because the TLSConfigChanged wraps context
|
// The background context is used because the TLSConfigChanged wraps context
|
||||||
// with timeout on its own and shuts down the server, which handles current
|
// with timeout on its own and shuts down the server, which handles current
|
||||||
// request. It is also should be done in a separate goroutine due to the
|
// request. It is also should be done in a separate goroutine due to the
|
||||||
// same reason.
|
// same reason.
|
||||||
if restartHTTPS {
|
if restartHTTPS {
|
||||||
go func() {
|
go m.web.tlsConfigChanged(context.Background(), &req.tlsConfigSettings)
|
||||||
m.web.tlsConfigChanged(context.Background(), req.tlsConfigSettings)
|
|
||||||
}()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -113,10 +113,13 @@ func TestValidateCertificates(t *testing.T) {
|
|||||||
// restores them once the test is complete.
|
// restores them once the test is complete.
|
||||||
//
|
//
|
||||||
// The global variables are:
|
// The global variables are:
|
||||||
// - [configuration.dns]
|
// - [configuration]
|
||||||
|
// - [homeContext.auth]
|
||||||
// - [homeContext.clients.storage]
|
// - [homeContext.clients.storage]
|
||||||
// - [homeContext.dnsServer]
|
// - [homeContext.dnsServer]
|
||||||
|
// - [homeContext.firstRun]
|
||||||
// - [homeContext.mux]
|
// - [homeContext.mux]
|
||||||
|
// - [homeContext.web]
|
||||||
//
|
//
|
||||||
// TODO(s.chzhen): Remove this once the TLS manager no longer accesses global
|
// TODO(s.chzhen): Remove this once the TLS manager no longer accesses global
|
||||||
// variables. Make tests that use this helper concurrent.
|
// variables. Make tests that use this helper concurrent.
|
||||||
@@ -124,15 +127,21 @@ func storeGlobals(tb testing.TB) {
|
|||||||
tb.Helper()
|
tb.Helper()
|
||||||
|
|
||||||
prevConfig := config
|
prevConfig := config
|
||||||
|
auth := globalContext.auth
|
||||||
storage := globalContext.clients.storage
|
storage := globalContext.clients.storage
|
||||||
dnsServer := globalContext.dnsServer
|
dnsServer := globalContext.dnsServer
|
||||||
|
firstRun := globalContext.firstRun
|
||||||
mux := globalContext.mux
|
mux := globalContext.mux
|
||||||
|
web := globalContext.web
|
||||||
|
|
||||||
tb.Cleanup(func() {
|
tb.Cleanup(func() {
|
||||||
config = prevConfig
|
config = prevConfig
|
||||||
|
globalContext.auth = auth
|
||||||
globalContext.clients.storage = storage
|
globalContext.clients.storage = storage
|
||||||
globalContext.dnsServer = dnsServer
|
globalContext.dnsServer = dnsServer
|
||||||
|
globalContext.firstRun = firstRun
|
||||||
globalContext.mux = mux
|
globalContext.mux = mux
|
||||||
|
globalContext.web = web
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -204,6 +213,8 @@ func assertCertSerialNumber(tb testing.TB, conf *tlsConfigSettings, wantSN int64
|
|||||||
func TestTLSManager_Reload(t *testing.T) {
|
func TestTLSManager_Reload(t *testing.T) {
|
||||||
storeGlobals(t)
|
storeGlobals(t)
|
||||||
|
|
||||||
|
config.DNS.Port = 0
|
||||||
|
|
||||||
var (
|
var (
|
||||||
logger = slogutil.NewDiscardLogger()
|
logger = slogutil.NewDiscardLogger()
|
||||||
ctx = testutil.ContextWithTimeout(t, testTimeout)
|
ctx = testutil.ContextWithTimeout(t, testTimeout)
|
||||||
@@ -239,11 +250,9 @@ func TestTLSManager_Reload(t *testing.T) {
|
|||||||
logger: logger,
|
logger: logger,
|
||||||
configModified: func() {},
|
configModified: func() {},
|
||||||
tlsSettings: tlsConfigSettings{
|
tlsSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
TLSConfig: dnsforward.TLSConfig{
|
CertificatePath: certPath,
|
||||||
CertificatePath: certPath,
|
PrivateKeyPath: keyPath,
|
||||||
PrivateKeyPath: keyPath,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
servePlainDNS: false,
|
servePlainDNS: false,
|
||||||
})
|
})
|
||||||
@@ -254,8 +263,7 @@ func TestTLSManager_Reload(t *testing.T) {
|
|||||||
|
|
||||||
m.setWebAPI(web)
|
m.setWebAPI(web)
|
||||||
|
|
||||||
conf := &tlsConfigSettings{}
|
conf := m.config()
|
||||||
m.WriteDiskConfig(conf)
|
|
||||||
assertCertSerialNumber(t, conf, snBefore)
|
assertCertSerialNumber(t, conf, snBefore)
|
||||||
|
|
||||||
certDER, key = newCertAndKey(t, snAfter)
|
certDER, key = newCertAndKey(t, snAfter)
|
||||||
@@ -263,7 +271,11 @@ func TestTLSManager_Reload(t *testing.T) {
|
|||||||
|
|
||||||
m.reload(ctx)
|
m.reload(ctx)
|
||||||
|
|
||||||
m.WriteDiskConfig(conf)
|
// The [tlsManager.reload] method will start the DNS server and it should be
|
||||||
|
// stopped after the test ends.
|
||||||
|
testutil.CleanupAndRequireSuccess(t, globalContext.dnsServer.Stop)
|
||||||
|
|
||||||
|
conf = m.config()
|
||||||
assertCertSerialNumber(t, conf, snAfter)
|
assertCertSerialNumber(t, conf, snAfter)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -278,11 +290,9 @@ func TestTLSManager_HandleTLSStatus(t *testing.T) {
|
|||||||
logger: logger,
|
logger: logger,
|
||||||
configModified: func() {},
|
configModified: func() {},
|
||||||
tlsSettings: tlsConfigSettings{
|
tlsSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
TLSConfig: dnsforward.TLSConfig{
|
CertificateChain: string(testCertChainData),
|
||||||
CertificateChain: string(testCertChainData),
|
PrivateKey: string(testPrivateKeyData),
|
||||||
PrivateKey: string(testPrivateKeyData),
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
servePlainDNS: false,
|
servePlainDNS: false,
|
||||||
})
|
})
|
||||||
@@ -342,47 +352,49 @@ func TestValidateTLSSettings(t *testing.T) {
|
|||||||
busyUDPPort := udpAddr.Port
|
busyUDPPort := udpAddr.Port
|
||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
setts tlsConfigSettingsExt
|
|
||||||
name string
|
name string
|
||||||
wantErr string
|
wantErr string
|
||||||
|
setts tlsConfigSettingsExt
|
||||||
}{{
|
}{{
|
||||||
name: "basic",
|
name: "basic",
|
||||||
setts: tlsConfigSettingsExt{},
|
|
||||||
wantErr: "",
|
wantErr: "",
|
||||||
|
setts: tlsConfigSettingsExt{},
|
||||||
}, {
|
}, {
|
||||||
|
name: "disabled_all",
|
||||||
|
wantErr: "plain DNS is required in case encryption protocols are disabled",
|
||||||
setts: tlsConfigSettingsExt{
|
setts: tlsConfigSettingsExt{
|
||||||
ServePlainDNS: aghalg.NBFalse,
|
ServePlainDNS: aghalg.NBFalse,
|
||||||
},
|
},
|
||||||
name: "disabled_all",
|
|
||||||
wantErr: "plain DNS is required in case encryption protocols are disabled",
|
|
||||||
}, {
|
}, {
|
||||||
|
name: "busy_https_port",
|
||||||
|
wantErr: fmt.Sprintf("port %d for HTTPS is not available", busyTCPPort),
|
||||||
setts: tlsConfigSettingsExt{
|
setts: tlsConfigSettingsExt{
|
||||||
tlsConfigSettings: tlsConfigSettings{
|
tlsConfigSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
PortHTTPS: uint16(busyTCPPort),
|
PortHTTPS: uint16(busyTCPPort),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
name: "busy_https_port",
|
|
||||||
wantErr: fmt.Sprintf("port %d for HTTPS is not available", busyTCPPort),
|
|
||||||
}, {
|
}, {
|
||||||
|
name: "busy_dot_port",
|
||||||
|
wantErr: fmt.Sprintf("port %d for DNS-over-TLS is not available", busyTCPPort),
|
||||||
setts: tlsConfigSettingsExt{
|
setts: tlsConfigSettingsExt{
|
||||||
tlsConfigSettings: tlsConfigSettings{
|
tlsConfigSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
PortDNSOverTLS: uint16(busyTCPPort),
|
PortDNSOverTLS: uint16(busyTCPPort),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
name: "busy_dot_port",
|
|
||||||
wantErr: fmt.Sprintf("port %d for DNS-over-TLS is not available", busyTCPPort),
|
|
||||||
}, {
|
}, {
|
||||||
|
name: "busy_doq_port",
|
||||||
|
wantErr: fmt.Sprintf("port %d for DNS-over-QUIC is not available", busyUDPPort),
|
||||||
setts: tlsConfigSettingsExt{
|
setts: tlsConfigSettingsExt{
|
||||||
tlsConfigSettings: tlsConfigSettings{
|
tlsConfigSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
PortDNSOverQUIC: uint16(busyUDPPort),
|
PortDNSOverQUIC: uint16(busyUDPPort),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
name: "busy_doq_port",
|
|
||||||
wantErr: fmt.Sprintf("port %d for DNS-over-QUIC is not available", busyUDPPort),
|
|
||||||
}, {
|
}, {
|
||||||
|
name: "duplicate_port",
|
||||||
|
wantErr: "validating tcp ports: duplicated values: [4433]",
|
||||||
setts: tlsConfigSettingsExt{
|
setts: tlsConfigSettingsExt{
|
||||||
tlsConfigSettings: tlsConfigSettings{
|
tlsConfigSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
@@ -390,8 +402,6 @@ func TestValidateTLSSettings(t *testing.T) {
|
|||||||
PortDNSOverTLS: 4433,
|
PortDNSOverTLS: 4433,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
name: "duplicate_port",
|
|
||||||
wantErr: "validating tcp ports: duplicated values: [4433]",
|
|
||||||
}}
|
}}
|
||||||
|
|
||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
@@ -417,11 +427,9 @@ func TestTLSManager_HandleTLSValidate(t *testing.T) {
|
|||||||
logger: logger,
|
logger: logger,
|
||||||
configModified: func() {},
|
configModified: func() {},
|
||||||
tlsSettings: tlsConfigSettings{
|
tlsSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
TLSConfig: dnsforward.TLSConfig{
|
CertificateChain: string(testCertChainData),
|
||||||
CertificateChain: string(testCertChainData),
|
PrivateKey: string(testPrivateKeyData),
|
||||||
PrivateKey: string(testPrivateKeyData),
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
servePlainDNS: false,
|
servePlainDNS: false,
|
||||||
})
|
})
|
||||||
@@ -434,11 +442,9 @@ func TestTLSManager_HandleTLSValidate(t *testing.T) {
|
|||||||
|
|
||||||
setts := &tlsConfigSettingsExt{
|
setts := &tlsConfigSettingsExt{
|
||||||
tlsConfigSettings: tlsConfigSettings{
|
tlsConfigSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
TLSConfig: dnsforward.TLSConfig{
|
CertificateChain: base64.StdEncoding.EncodeToString(testCertChainData),
|
||||||
CertificateChain: base64.StdEncoding.EncodeToString(testCertChainData),
|
PrivateKey: base64.StdEncoding.EncodeToString(testPrivateKeyData),
|
||||||
PrivateKey: base64.StdEncoding.EncodeToString(testPrivateKeyData),
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -476,6 +482,7 @@ func TestTLSManager_HandleTLSConfigure(t *testing.T) {
|
|||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
err = globalContext.dnsServer.Prepare(&dnsforward.ServerConfig{
|
err = globalContext.dnsServer.Prepare(&dnsforward.ServerConfig{
|
||||||
|
TLSConf: &dnsforward.TLSConfig{},
|
||||||
Config: dnsforward.Config{
|
Config: dnsforward.Config{
|
||||||
UpstreamMode: dnsforward.UpstreamModeLoadBalance,
|
UpstreamMode: dnsforward.UpstreamModeLoadBalance,
|
||||||
EDNSClientSubnet: &dnsforward.EDNSClientSubnet{Enabled: false},
|
EDNSClientSubnet: &dnsforward.EDNSClientSubnet{Enabled: false},
|
||||||
@@ -511,11 +518,9 @@ func TestTLSManager_HandleTLSConfigure(t *testing.T) {
|
|||||||
logger: logger,
|
logger: logger,
|
||||||
configModified: func() {},
|
configModified: func() {},
|
||||||
tlsSettings: tlsConfigSettings{
|
tlsSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
TLSConfig: dnsforward.TLSConfig{
|
CertificatePath: certPath,
|
||||||
CertificatePath: certPath,
|
PrivateKeyPath: keyPath,
|
||||||
PrivateKeyPath: keyPath,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
servePlainDNS: true,
|
servePlainDNS: true,
|
||||||
})
|
})
|
||||||
@@ -526,19 +531,16 @@ func TestTLSManager_HandleTLSConfigure(t *testing.T) {
|
|||||||
|
|
||||||
m.setWebAPI(web)
|
m.setWebAPI(web)
|
||||||
|
|
||||||
conf := &tlsConfigSettings{}
|
conf := m.config()
|
||||||
m.WriteDiskConfig(conf)
|
|
||||||
assertCertSerialNumber(t, conf, wantSerialNumber)
|
assertCertSerialNumber(t, conf, wantSerialNumber)
|
||||||
|
|
||||||
// Prepare a request with the new TLS configuration.
|
// Prepare a request with the new TLS configuration.
|
||||||
setts := &tlsConfigSettingsExt{
|
setts := &tlsConfigSettingsExt{
|
||||||
tlsConfigSettings: tlsConfigSettings{
|
tlsConfigSettings: tlsConfigSettings{
|
||||||
Enabled: true,
|
Enabled: true,
|
||||||
PortHTTPS: 4433,
|
PortHTTPS: 4433,
|
||||||
TLSConfig: dnsforward.TLSConfig{
|
CertificateChain: base64.StdEncoding.EncodeToString(testCertChainData),
|
||||||
CertificateChain: base64.StdEncoding.EncodeToString(testCertChainData),
|
PrivateKey: base64.StdEncoding.EncodeToString(testPrivateKeyData),
|
||||||
PrivateKey: base64.StdEncoding.EncodeToString(testPrivateKeyData),
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -157,8 +157,8 @@ func newWebAPI(ctx context.Context, conf *webConfig) (w *webAPI) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// tlsConfigChanged updates the TLS configuration and restarts the HTTPS server
|
// tlsConfigChanged updates the TLS configuration and restarts the HTTPS server
|
||||||
// if necessary.
|
// if necessary. tlsConf must not be nil.
|
||||||
func (web *webAPI) tlsConfigChanged(ctx context.Context, tlsConf tlsConfigSettings) {
|
func (web *webAPI) tlsConfigChanged(ctx context.Context, tlsConf *tlsConfigSettings) {
|
||||||
defer slogutil.RecoverAndExit(ctx, web.logger, osutil.ExitCodeFailure)
|
defer slogutil.RecoverAndExit(ctx, web.logger, osutil.ExitCodeFailure)
|
||||||
|
|
||||||
web.logger.DebugContext(ctx, "applying new tls configuration")
|
web.logger.DebugContext(ctx, "applying new tls configuration")
|
||||||
|
|||||||
@@ -64,7 +64,7 @@ type Entry struct {
|
|||||||
Domain string
|
Domain string
|
||||||
|
|
||||||
// UpstreamStats contains the DNS query statistics for both the upstream and
|
// UpstreamStats contains the DNS query statistics for both the upstream and
|
||||||
// fallback DNS servers.
|
// fallback DNS servers. Don't modify items in the slice.
|
||||||
UpstreamStats []*proxy.UpstreamStatistics
|
UpstreamStats []*proxy.UpstreamStatistics
|
||||||
|
|
||||||
// Result is the result of processing the request.
|
// Result is the result of processing the request.
|
||||||
|
|||||||
@@ -980,7 +980,8 @@
|
|||||||
- 'clients'
|
- 'clients'
|
||||||
'operationId': 'clientsSearch'
|
'operationId': 'clientsSearch'
|
||||||
'summary': >
|
'summary': >
|
||||||
Get information about clients by their IP addresses, CIDRs, MAC addresses, or ClientIDs.
|
Retrieve information about clients by performing an exact match search
|
||||||
|
using IP addresses, CIDRs, MAC addresses, or ClientIDs.
|
||||||
'requestBody':
|
'requestBody':
|
||||||
'content':
|
'content':
|
||||||
'application/json':
|
'application/json':
|
||||||
|
|||||||
@@ -119,4 +119,5 @@ $sudo_cmd docker "$debug_flags" \
|
|||||||
--build-arg VERSION="$version" \
|
--build-arg VERSION="$version" \
|
||||||
--output "$docker_output" \
|
--output "$docker_output" \
|
||||||
--platform "$docker_platforms" \
|
--platform "$docker_platforms" \
|
||||||
|
--progress 'plain' \
|
||||||
$docker_version_tag $docker_channel_tag -f ./docker/Dockerfile .
|
$docker_version_tag $docker_channel_tag -f ./docker/Dockerfile .
|
||||||
|
|||||||
@@ -199,6 +199,7 @@ run_linter gocognit --over='10' \
|
|||||||
./internal/aghhttp/ \
|
./internal/aghhttp/ \
|
||||||
./internal/aghrenameio/ \
|
./internal/aghrenameio/ \
|
||||||
./internal/aghtest/ \
|
./internal/aghtest/ \
|
||||||
|
./internal/aghuser/ \
|
||||||
./internal/arpdb/ \
|
./internal/arpdb/ \
|
||||||
./internal/client/ \
|
./internal/client/ \
|
||||||
./internal/configmigrate/ \
|
./internal/configmigrate/ \
|
||||||
@@ -250,6 +251,7 @@ run_linter fieldalignment \
|
|||||||
./internal/aghrenameio/ \
|
./internal/aghrenameio/ \
|
||||||
./internal/aghtest/ \
|
./internal/aghtest/ \
|
||||||
./internal/aghtls/ \
|
./internal/aghtls/ \
|
||||||
|
./internal/aghuser/ \
|
||||||
./internal/arpdb/ \
|
./internal/arpdb/ \
|
||||||
./internal/client/ \
|
./internal/client/ \
|
||||||
./internal/configmigrate/ \
|
./internal/configmigrate/ \
|
||||||
@@ -280,6 +282,7 @@ run_linter gosec --exclude G115 --quiet \
|
|||||||
./internal/aghos/ \
|
./internal/aghos/ \
|
||||||
./internal/aghrenameio/ \
|
./internal/aghrenameio/ \
|
||||||
./internal/aghtest/ \
|
./internal/aghtest/ \
|
||||||
|
./internal/aghuser/ \
|
||||||
./internal/arpdb/ \
|
./internal/arpdb/ \
|
||||||
./internal/client/ \
|
./internal/client/ \
|
||||||
./internal/configmigrate/ \
|
./internal/configmigrate/ \
|
||||||
|
|||||||
Reference in New Issue
Block a user