mirror of
https://github.com/telemt/telemt.git
synced 2026-06-24 11:51:10 +03:00
Compare commits
31 Commits
0b580eccd3
...
3.4.0
| Author | SHA1 | Date | |
|---|---|---|---|
| 32d5cee01c | |||
| 3a17901e83 | |||
| 902a4e83cf | |||
| 696316f919 | |||
| d7a0319696 | |||
| 3fefcdd11f | |||
| 57dca639f0 | |||
| 13f86062f4 | |||
| 9303c7854a | |||
| 8267149b53 | |||
| 30fab00bfd | |||
| afc07345f5 | |||
| a965b38bd4 | |||
| f0ebbac338 | |||
| 286662fc51 | |||
| c5390baaf1 | |||
| 1cd1e96079 | |||
| 2b995c31b0 | |||
| 442320302d | |||
| ac0dde567b | |||
| b2fe9b78d8 | |||
| f039ce1827 | |||
| abff2fd7fe | |||
| 5f5a3e3fa0 | |||
| f9e54ee739 | |||
| d477d6ee29 | |||
| 1383dfcbb1 | |||
| 107a7cc758 | |||
| 4f3193fdaa | |||
| d6be691c67 | |||
| 0b0be07a9c |
Generated
+1
-1
@@ -2780,7 +2780,7 @@ checksum = "7b2093cf4c8eb1e67749a6762251bc9cd836b6fc171623bd0a9d324d37af2417"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "telemt"
|
name = "telemt"
|
||||||
version = "3.3.39"
|
version = "3.4.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"aes",
|
"aes",
|
||||||
"anyhow",
|
"anyhow",
|
||||||
|
|||||||
+1
-1
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "telemt"
|
name = "telemt"
|
||||||
version = "3.3.39"
|
version = "3.4.0"
|
||||||
edition = "2024"
|
edition = "2024"
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
   [](https://t.me/telemtrs)
|
   [](https://t.me/telemtrs)
|
||||||
|
|
||||||
|
[🇷🇺 README на русском](https://github.com/telemt/telemt/blob/main/README.ru.md)
|
||||||
|
|
||||||
***Löst Probleme, bevor andere überhaupt wissen, dass sie existieren*** / ***It solves problems before others even realize they exist***
|
***Löst Probleme, bevor andere überhaupt wissen, dass sie existieren*** / ***It solves problems before others even realize they exist***
|
||||||
|
|
||||||
> [!NOTE]
|
> [!NOTE]
|
||||||
@@ -25,6 +27,7 @@ curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh
|
|||||||
- [Quick Start Guide](docs/Quick_start/QUICK_START_GUIDE.en.md)
|
- [Quick Start Guide](docs/Quick_start/QUICK_START_GUIDE.en.md)
|
||||||
- [Инструкция по быстрому запуску](docs/Quick_start/QUICK_START_GUIDE.ru.md)
|
- [Инструкция по быстрому запуску](docs/Quick_start/QUICK_START_GUIDE.ru.md)
|
||||||
|
|
||||||
|
## Features
|
||||||
Our implementation of **TLS-fronting** is one of the most deeply debugged, focused, advanced and *almost* **"behaviorally consistent to real"**: we are confident we have it right - [see evidence on our validation and traces](docs/FAQ.en.md#recognizability-for-dpi-and-crawler)
|
Our implementation of **TLS-fronting** is one of the most deeply debugged, focused, advanced and *almost* **"behaviorally consistent to real"**: we are confident we have it right - [see evidence on our validation and traces](docs/FAQ.en.md#recognizability-for-dpi-and-crawler)
|
||||||
|
|
||||||
Our ***Middle-End Pool*** is fastest by design in standard scenarios, compared to other implementations of connecting to the Middle-End Proxy: non dramatically, but usual
|
Our ***Middle-End Pool*** is fastest by design in standard scenarios, compared to other implementations of connecting to the Middle-End Proxy: non dramatically, but usual
|
||||||
|
|||||||
+21
-2
@@ -54,7 +54,6 @@ curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh
|
|||||||
- [FAQ EN](docs/FAQ.en.md)
|
- [FAQ EN](docs/FAQ.en.md)
|
||||||
|
|
||||||
## Сборка
|
## Сборка
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Клонируйте репозиторий
|
# Клонируйте репозиторий
|
||||||
git clone https://github.com/telemt/telemt
|
git clone https://github.com/telemt/telemt
|
||||||
@@ -63,7 +62,6 @@ cd telemt
|
|||||||
# Начните процесс сборки
|
# Начните процесс сборки
|
||||||
cargo build --release
|
cargo build --release
|
||||||
|
|
||||||
# Устройства с небольшим объёмом оперативной памяти (1 ГБ, например NanoPi Neo3 / Raspberry Pi Zero 2):
|
|
||||||
# В текущем release-профиле используется lto = "fat" для максимальной оптимизации (см. Cargo.toml).
|
# В текущем release-профиле используется lto = "fat" для максимальной оптимизации (см. Cargo.toml).
|
||||||
# На системах с малым объёмом RAM (~1 ГБ) можно переопределить это значение на "thin".
|
# На системах с малым объёмом RAM (~1 ГБ) можно переопределить это значение на "thin".
|
||||||
|
|
||||||
@@ -87,4 +85,25 @@ telemt config.toml
|
|||||||
- Безопасность памяти;
|
- Безопасность памяти;
|
||||||
- Асинхронная архитектура Tokio.
|
- Асинхронная архитектура Tokio.
|
||||||
|
|
||||||
|
## Поддержать Telemt
|
||||||
|
|
||||||
|
Telemt — это бесплатное программное обеспечение с открытым исходным кодом, разработанное в свободное время.
|
||||||
|
Если оно оказалось вам полезным, вы можете поддержать дальнейшую разработку.
|
||||||
|
|
||||||
|
Принимаемые криптовалюты (BTC, ETH, USDT, 350+ и другие):
|
||||||
|
|
||||||
|
<p align="center">
|
||||||
|
<a href="https://nowpayments.io/donation?api_key=2bf1afd2-abc2-49f9-a012-f1e715b37223" target="_blank" rel="noreferrer noopener">
|
||||||
|
<img src="https://nowpayments.io/images/embeds/donation-button-white.svg" alt="Cryptocurrency & Bitcoin donation button by NOWPayments" height="80">
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
Monero (XMR) напрямую:
|
||||||
|
|
||||||
|
```
|
||||||
|
8Bk4tZEYPQWSypeD2hrUXG2rKbAKF16GqEN942ZdAP5cFdSqW6h4DwkP5cJMAdszzuPeHeHZPTyjWWFwzeFdjuci3ktfMoB
|
||||||
|
```
|
||||||
|
|
||||||
|
Все пожертвования пойдут на инфраструктуру, разработку и исследования.
|
||||||
|
|
||||||

|

|
||||||
|
|||||||
+8
-5
@@ -32,13 +32,13 @@ show = "*"
|
|||||||
port = 443
|
port = 443
|
||||||
# proxy_protocol = false # Enable if behind HAProxy/nginx with PROXY protocol
|
# proxy_protocol = false # Enable if behind HAProxy/nginx with PROXY protocol
|
||||||
# metrics_port = 9090
|
# metrics_port = 9090
|
||||||
# metrics_listen = "0.0.0.0:9090" # Listen address for metrics (overrides metrics_port)
|
# metrics_listen = "127.0.0.1:9090" # Listen address for metrics (overrides metrics_port)
|
||||||
# metrics_whitelist = ["127.0.0.1", "::1", "0.0.0.0/0"]
|
# metrics_whitelist = ["127.0.0.1/32", "::1/128"]
|
||||||
|
|
||||||
[server.api]
|
[server.api]
|
||||||
enabled = true
|
enabled = true
|
||||||
listen = "0.0.0.0:9091"
|
listen = "127.0.0.1:9091"
|
||||||
whitelist = ["127.0.0.0/8"]
|
whitelist = ["127.0.0.1/32", "::1/128"]
|
||||||
minimal_runtime_enabled = false
|
minimal_runtime_enabled = false
|
||||||
minimal_runtime_cache_ttl_ms = 1000
|
minimal_runtime_cache_ttl_ms = 1000
|
||||||
|
|
||||||
@@ -48,9 +48,12 @@ ip = "0.0.0.0"
|
|||||||
|
|
||||||
# === Anti-Censorship & Masking ===
|
# === Anti-Censorship & Masking ===
|
||||||
[censorship]
|
[censorship]
|
||||||
|
# Fake-TLS / SNI masking domain used in generated ee-links.
|
||||||
|
# Changing tls_domain invalidates previously generated TLS links.
|
||||||
tls_domain = "petrovich.ru"
|
tls_domain = "petrovich.ru"
|
||||||
|
|
||||||
mask = true
|
mask = true
|
||||||
tls_emulation = true # Fetch real cert lengths and emulate TLS records
|
tls_emulation = true # Fetch real cert lengths and emulate TLS records
|
||||||
tls_front_dir = "tlsfront" # Cache directory for TLS emulation
|
tls_front_dir = "tlsfront" # Cache directory for TLS emulation
|
||||||
|
|
||||||
[access.users]
|
[access.users]
|
||||||
|
|||||||
@@ -9,12 +9,12 @@ API runtime is configured in `[server.api]`.
|
|||||||
|
|
||||||
| Field | Type | Default | Description |
|
| Field | Type | Default | Description |
|
||||||
| --- | --- | --- | --- |
|
| --- | --- | --- | --- |
|
||||||
| `enabled` | `bool` | `false` | Enables REST API listener. |
|
| `enabled` | `bool` | `true` | Enables REST API listener. |
|
||||||
| `listen` | `string` (`IP:PORT`) | `127.0.0.1:9091` | API bind address. |
|
| `listen` | `string` (`IP:PORT`) | `0.0.0.0:9091` | API bind address. |
|
||||||
| `whitelist` | `CIDR[]` | `127.0.0.1/32, ::1/128` | Source IP allowlist. Empty list means allow all. |
|
| `whitelist` | `CIDR[]` | `127.0.0.0/8` | Source IP allowlist. Empty list means allow all. |
|
||||||
| `auth_header` | `string` | `""` | Exact value for `Authorization` header. Empty disables header auth. |
|
| `auth_header` | `string` | `""` | Exact value for `Authorization` header. Empty disables header auth. |
|
||||||
| `request_body_limit_bytes` | `usize` | `65536` | Maximum request body size. Must be `> 0`. |
|
| `request_body_limit_bytes` | `usize` | `65536` | Maximum request body size. Must be `> 0`. |
|
||||||
| `minimal_runtime_enabled` | `bool` | `false` | Enables runtime snapshot endpoints requiring ME pool read-lock aggregation. |
|
| `minimal_runtime_enabled` | `bool` | `true` | Enables runtime snapshot endpoints requiring ME pool read-lock aggregation. |
|
||||||
| `minimal_runtime_cache_ttl_ms` | `u64` | `1000` | Cache TTL for minimal snapshots. `0` disables cache; valid range is `[0, 60000]`. |
|
| `minimal_runtime_cache_ttl_ms` | `u64` | `1000` | Cache TTL for minimal snapshots. `0` disables cache; valid range is `[0, 60000]`. |
|
||||||
| `runtime_edge_enabled` | `bool` | `false` | Enables runtime edge endpoints with cached aggregation payloads. |
|
| `runtime_edge_enabled` | `bool` | `false` | Enables runtime edge endpoints with cached aggregation payloads. |
|
||||||
| `runtime_edge_cache_ttl_ms` | `u64` | `1000` | Cache TTL for runtime edge summary payloads. `0` disables cache. |
|
| `runtime_edge_cache_ttl_ms` | `u64` | `1000` | Cache TTL for runtime edge summary payloads. `0` disables cache. |
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
+3072
-3331
File diff suppressed because it is too large
Load Diff
+25
-3
@@ -36,8 +36,11 @@ hello2 = "ad_tag2"
|
|||||||
On April 1, 2026, we became aware of a method for detecting MTProxy Fake-TLS,
|
On April 1, 2026, we became aware of a method for detecting MTProxy Fake-TLS,
|
||||||
based on the ECH extension and the ordering of cipher suites,
|
based on the ECH extension and the ordering of cipher suites,
|
||||||
as well as an overall unique JA3/JA4 fingerprint
|
as well as an overall unique JA3/JA4 fingerprint
|
||||||
that does not occur in modern browsers:
|
that does not occur in modern browsers.
|
||||||
we have already submitted initial changes to the Telegram Desktop developers and are working on updates for other clients.
|
|
||||||
|
> [!IMPORTANT]
|
||||||
|
> TLS fingerprint has been fixed in latest version of clients for Desktop / Android / iOS.
|
||||||
|
> Please update your client for MTProxy Fake-TLS to work correctly.
|
||||||
|
|
||||||
- We consider this a breakthrough aspect, which has no stable analogues today
|
- We consider this a breakthrough aspect, which has no stable analogues today
|
||||||
- Based on this: if `telemt` configured correctly, **TLS mode is completely identical to real-life handshake + communication** with a specified host
|
- Based on this: if `telemt` configured correctly, **TLS mode is completely identical to real-life handshake + communication** with a specified host
|
||||||
@@ -154,6 +157,24 @@ Keep-Alive: timeout=60
|
|||||||
### Why do you need a middle proxy (ME)
|
### Why do you need a middle proxy (ME)
|
||||||
https://github.com/telemt/telemt/discussions/167
|
https://github.com/telemt/telemt/discussions/167
|
||||||
|
|
||||||
|
## How clients interact with Telegram DCs
|
||||||
|
When you register a Telegram account, it gets permanently bound to one of Telegram's data centers (DCs).
|
||||||
|
It is deciced beforehand by Telegram based on the phone number's region.
|
||||||
|
This DC becomes your **home DC**: all content you upload (photos, videos, files, messages) is stored there.
|
||||||
|
Your client authenticates on it with every connection.
|
||||||
|
|
||||||
|
For example, if your account is registered on **DC2**, your client will always connect to DC2 first.
|
||||||
|
When you open a chat with another user whose home DC is **DC5**, your client opens an additional connection to DC5 to download their media.
|
||||||
|
Those cross-DC requests are normal and happen constantly.
|
||||||
|
|
||||||
|
> [!WARNING]
|
||||||
|
> Because every session is anchored to your home DC, an outage there causes other DCs to be unavaliable.
|
||||||
|
> If your home DC is DC2 and DC2 goes down, you **cannot** reach DC5 even though DC5 itself is perfectly healthy.
|
||||||
|
> The client has no valid session to route the request through.
|
||||||
|
|
||||||
|
This is also why an MTProxy only needs to reach Telegram's DC infrastructure as a whole.
|
||||||
|
The proxy itself doesn't care which DC your account lives on. The client negotiates the correct DC through the proxy after connecting.
|
||||||
|
|
||||||
### How many people can use one link
|
### How many people can use one link
|
||||||
By default, an unlimited number of people can use a single link.
|
By default, an unlimited number of people can use a single link.
|
||||||
However, you can limit the number of unique IP addresses for each user:
|
However, you can limit the number of unique IP addresses for each user:
|
||||||
@@ -161,7 +182,8 @@ However, you can limit the number of unique IP addresses for each user:
|
|||||||
[access.user_max_unique_ips]
|
[access.user_max_unique_ips]
|
||||||
hello = 1
|
hello = 1
|
||||||
```
|
```
|
||||||
This parameter sets the maximum number of unique IP addresses from which a single link can be used simultaneously. If the first user disconnects, a second one can connect. At the same time, multiple users can connect from a single IP address simultaneously (for example, devices on the same Wi-Fi network).
|
This parameter sets the maximum number of unique IP addresses from which a single link can be used simultaneously. If the first user disconnects, a second one can connect.
|
||||||
|
At the same time, multiple users can connect from a single IP address simultaneously (for example, devices on the same Wi-Fi network).
|
||||||
|
|
||||||
### How to create multiple different links
|
### How to create multiple different links
|
||||||
1. Generate the required number of secrets using the command: `openssl rand -hex 16`.
|
1. Generate the required number of secrets using the command: `openssl rand -hex 16`.
|
||||||
|
|||||||
+22
-2
@@ -33,9 +33,12 @@ hello = "ad_tag"
|
|||||||
hello2 = "ad_tag2"
|
hello2 = "ad_tag2"
|
||||||
```
|
```
|
||||||
## Распознаваемость для DPI и сканеров
|
## Распознаваемость для DPI и сканеров
|
||||||
|
1 апреля 2026 года нам стало известно о методе обнаружения MTProxy Fake-TLS, основанном на расширении ECH и порядке набора шифров,
|
||||||
|
а также об общем уникальном отпечатке JA3/JA4, который не встречается в современных браузерах.
|
||||||
|
|
||||||
1 апреля 2026 года нам стало известно о методе обнаружения MTProxy Fake-TLS, основанном на расширении ECH и порядке набора шифров,
|
> [!IMPORTANT]
|
||||||
а также об общем уникальном отпечатке JA3/JA4, который не встречается в современных браузерах: мы уже отправили первоначальные изменения разработчикам Telegram Desktop и работаем над обновлениями для других клиентов.
|
> Проблема с TLS отпечатком исправлена в последних версиях клиентов Telegram для Desktop / Android / iOS.
|
||||||
|
> Обновите свой клиент для корректной работы с MTProxy Fake-TLS!
|
||||||
|
|
||||||
- Мы считаем это прорывом, которому на сегодняшний день нет стабильных аналогов;
|
- Мы считаем это прорывом, которому на сегодняшний день нет стабильных аналогов;
|
||||||
- Исходя из этого: если `telemt` настроен правильно, **режим TLS полностью идентичен реальному «рукопожатию» + обмену данными** с указанным хостом;
|
- Исходя из этого: если `telemt` настроен правильно, **режим TLS полностью идентичен реальному «рукопожатию» + обмену данными** с указанным хостом;
|
||||||
@@ -152,6 +155,23 @@ Keep-Alive: timeout=60
|
|||||||
## Зачем нужен middle proxy (ME)
|
## Зачем нужен middle proxy (ME)
|
||||||
https://github.com/telemt/telemt/discussions/167
|
https://github.com/telemt/telemt/discussions/167
|
||||||
|
|
||||||
|
## Как клиенты взаимодействуют с дата-центрами Telegram
|
||||||
|
При регистрации аккаунта Telegram он навсегда привязывается к одному из дата-центров (DC).
|
||||||
|
Telegram заранее определяет к какому DC привязать аккаунт исходя из региона, к которому относиться номер телефона.
|
||||||
|
Этот DC становится вашим **домашним**: именно там хранится весь контент, который вы загружаете (фото, видео, файлы, сообщения).
|
||||||
|
И именно на нем клиент авторизуется при каждом подключении.
|
||||||
|
|
||||||
|
Например, если ваш аккаунт зарегистрирован на **DC2**, клиент всегда будет подключаться в первую очередь к DC2.
|
||||||
|
Когда вы открываете переписку с пользователем, чей домашний DC — **DC5**, клиент устанавливает доп. соединение с DC5, чтобы загрузить его контент.
|
||||||
|
Такие кросс-запросы к DC — это нормальная часть работы Telegram.
|
||||||
|
|
||||||
|
> [!WARNING]
|
||||||
|
> Поскольку аккаунт всегда привязан к домашнему DC, при его падении контент с других DC будет недоступен.
|
||||||
|
> Если ваш домашний DC — DC2, и DC2 лежит, вы **не сможете** достучаться и до DC5, даже если сам DC5 полностью исправен.
|
||||||
|
> У клиента просто нет валидной сессии, через которую можно было бы направить запрос.
|
||||||
|
|
||||||
|
По той же причине MTProxy достаточно иметь доступ к инфраструктуре Telegram в целом.
|
||||||
|
Cамому MTProxy всё равно, на каком DC живёт ваш аккаунт. Клиент cам договаривается о нужном DC через прокси уже после подключения.
|
||||||
|
|
||||||
## Что такое dd и ee в контексте MTProxy?
|
## Что такое dd и ee в контексте MTProxy?
|
||||||
|
|
||||||
|
|||||||
@@ -27,7 +27,8 @@ cargo build --release
|
|||||||
./target/release/telemt --version
|
./target/release/telemt --version
|
||||||
```
|
```
|
||||||
|
|
||||||
For low-RAM systems, this repository already uses `lto = "thin"` in release profile.
|
For low-RAM systems, note that this repository currently uses `lto = "fat"` in release profile.
|
||||||
|
On constrained builders, a local override to `lto = "thin"` may be more practical.
|
||||||
|
|
||||||
## 3. Install binary and config
|
## 3. Install binary and config
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,36 @@
|
|||||||
|
# Installation Options
|
||||||
|
There are three options for installing Telemt:
|
||||||
|
- [Automated installation using a script](#very-quick-start).
|
||||||
|
- [Manual installation of Telemt as a service](#telemt-via-systemd).
|
||||||
|
- [Installation using Docker Compose](#telemt-via-docker-compose).
|
||||||
|
|
||||||
# Very quick start
|
# Very quick start
|
||||||
|
|
||||||
### One-command installation / update on re-run
|
### One-command installation / update on re-run
|
||||||
```bash
|
```bash
|
||||||
curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh
|
curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh
|
||||||
```
|
```
|
||||||
|
|
||||||
|
After starting, the script will prompt for:
|
||||||
|
- Your language (1 - English, 2 - Russian);
|
||||||
|
- Your TLS domain (press Enter for petrovich.ru).
|
||||||
|
|
||||||
|
The script checks if the port (default **443**) is free. If the port is already in use, installation will fail. You need to free up the port or use the **-p** flag with a different port to retry the installation.
|
||||||
|
|
||||||
|
To modify the script’s startup parameters, you can use the following flags:
|
||||||
|
- **-d, --domain** - TLS domain;
|
||||||
|
- **-p, --port** - server port (1–65535);
|
||||||
|
- **-s, --secret** - 32 hex secret;
|
||||||
|
- **-a, --ad-tag** - ad_tag;
|
||||||
|
- **-l, --lan**g - language (1/en or 2/ru);
|
||||||
|
|
||||||
|
Providing all options skips interactive prompts.
|
||||||
|
|
||||||
|
After completion, the script will provide a link for client connections:
|
||||||
|
```bash
|
||||||
|
tg://proxy?server=IP&port=PORT&secret=SECRET
|
||||||
|
```
|
||||||
|
|
||||||
### Installing a specific version
|
### Installing a specific version
|
||||||
```bash
|
```bash
|
||||||
curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh -s -- 3.3.39
|
curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh -s -- 3.3.39
|
||||||
@@ -110,15 +137,15 @@ show = "*"
|
|||||||
# === Server Binding ===
|
# === Server Binding ===
|
||||||
[server]
|
[server]
|
||||||
port = 443
|
port = 443
|
||||||
# proxy_protocol = false # Enable if behind HAProxy/nginx with PROXY protocol
|
# proxy_protocol = false # Enable if behind HAProxy/nginx with PROXY protocol
|
||||||
# metrics_port = 9090
|
# metrics_port = 9090
|
||||||
# metrics_listen = "0.0.0.0:9090" # Listen address for metrics (overrides metrics_port)
|
# metrics_listen = "127.0.0.1:9090" # Listen address for metrics (overrides metrics_port)
|
||||||
# metrics_whitelist = ["127.0.0.1", "::1", "0.0.0.0/0"]
|
# metrics_whitelist = ["127.0.0.1/32", "::1/128"]
|
||||||
|
|
||||||
[server.api]
|
[server.api]
|
||||||
enabled = true
|
enabled = true
|
||||||
listen = "0.0.0.0:9091"
|
listen = "127.0.0.1:9091"
|
||||||
whitelist = ["127.0.0.0/8"]
|
whitelist = ["127.0.0.1/32", "::1/128"]
|
||||||
minimal_runtime_enabled = false
|
minimal_runtime_enabled = false
|
||||||
minimal_runtime_cache_ttl_ms = 1000
|
minimal_runtime_cache_ttl_ms = 1000
|
||||||
|
|
||||||
@@ -128,9 +155,9 @@ ip = "0.0.0.0"
|
|||||||
|
|
||||||
# === Anti-Censorship & Masking ===
|
# === Anti-Censorship & Masking ===
|
||||||
[censorship]
|
[censorship]
|
||||||
tls_domain = "petrovich.ru"
|
tls_domain = "petrovich.ru" # Fake-TLS / SNI masking domain used in generated ee-links
|
||||||
mask = true
|
mask = true
|
||||||
tls_emulation = true # Fetch real cert lengths and emulate TLS records
|
tls_emulation = true # Fetch real cert lengths and emulate TLS records
|
||||||
tls_front_dir = "tlsfront" # Cache directory for TLS emulation
|
tls_front_dir = "tlsfront" # Cache directory for TLS emulation
|
||||||
|
|
||||||
[access.users]
|
[access.users]
|
||||||
@@ -141,9 +168,9 @@ hello = "00000000000000000000000000000000"
|
|||||||
then Ctrl+S -> Ctrl+X to save
|
then Ctrl+S -> Ctrl+X to save
|
||||||
|
|
||||||
> [!WARNING]
|
> [!WARNING]
|
||||||
> Replace the value of the hello parameter with the value you obtained in step 0.
|
> Replace the value of the `hello` parameter with the value you obtained in step 0.
|
||||||
> Additionally, change the value of the tls_domain parameter to a different website.
|
> Additionally, change the value of the `tls_domain` parameter to a different website.
|
||||||
> Changing the tls_domain parameter will break all links that use the old domain!
|
> Changing the `tls_domain` parameter will break all links that use the old domain!
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,35 @@
|
|||||||
|
# Варианты установки
|
||||||
|
Имеется три варианта установки Telemt:
|
||||||
|
- [Автоматизированная установка с помощью скрипта](#очень-быстрый-старт).
|
||||||
|
- [Ручная установка Telemt в качестве службы](#telemt-через-systemd-вручную).
|
||||||
|
- [Установка через Docker Compose](#telemt-через-docker-compose).
|
||||||
|
|
||||||
# Очень быстрый старт
|
# Очень быстрый старт
|
||||||
|
|
||||||
### Установка одной командой / обновление при повторном запуске
|
### Установка одной командой / обновление при повторном запуске
|
||||||
```bash
|
```bash
|
||||||
curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh
|
curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh
|
||||||
```
|
```
|
||||||
|
После запуска скрипт запросит:
|
||||||
|
- ваш язык (1 - English, 2 - Русский);
|
||||||
|
- ваш TLS-домен (нажмите Enter для petrovich.ru).
|
||||||
|
|
||||||
|
Во время установки скрипт проверяет, свободен ли порт (по умолчанию **443**). Если порт занят другим процессом - установка завершится с ошибкой. Для повторной установки необходимо освободить порт или указать другой через флаг **-p**.
|
||||||
|
|
||||||
|
Для изменения параметров запуска скрипта можно использовать следующие флаги:
|
||||||
|
- **-d, --domain** - TLS-домен;
|
||||||
|
- **-p, --port** - порт (1–65535);
|
||||||
|
- **-s, --secret** - секрет (32 hex символа);
|
||||||
|
- **-a, --ad-tag** - ad_tag;
|
||||||
|
- **-l, --lang** - язык (1/en или 2/ru).
|
||||||
|
|
||||||
|
Если заданы флаги для языка и домена, интерактивных вопросов не будет.
|
||||||
|
|
||||||
|
После завершения установки скрипт выдаст ссылку для подключения клиентов:
|
||||||
|
```bash
|
||||||
|
tg://proxy?server=IP&port=PORT&secret=SECRET
|
||||||
|
```
|
||||||
|
|
||||||
### Установка нужной версии
|
### Установка нужной версии
|
||||||
```bash
|
```bash
|
||||||
curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh -s -- 3.3.39
|
curl -fsSL https://raw.githubusercontent.com/telemt/telemt/main/install.sh | sh -s -- 3.3.39
|
||||||
@@ -103,22 +129,22 @@ tls = true
|
|||||||
[general.links]
|
[general.links]
|
||||||
show = "*"
|
show = "*"
|
||||||
# show = ["alice", "bob"] # Показывать ссылки только для alice и bob
|
# show = ["alice", "bob"] # Показывать ссылки только для alice и bob
|
||||||
# show = "*" # Показывать ссылки для всех пользователей
|
# show = "*" # Показывать ссылки для всех пользователей
|
||||||
# public_host = "proxy.example.com" # Хост (IP-адрес или домен) для ссылок tg://
|
# public_host = "proxy.example.com" # Хост (IP-адрес или домен) для ссылок tg://
|
||||||
# public_port = 443 # Порт для ссылок tg:// (по умолчанию: server.port)
|
# public_port = 443 # Порт для ссылок tg:// (по умолчанию: server.port)
|
||||||
|
|
||||||
# === Привязка сервера ===
|
# === Привязка сервера ===
|
||||||
[server]
|
[server]
|
||||||
port = 443
|
port = 443
|
||||||
# proxy_protocol = false # Включите, если сервер находится за HAProxy/nginx с протоколом PROXY
|
# proxy_protocol = false # Включите, если сервер находится за HAProxy/nginx с протоколом PROXY
|
||||||
# metrics_port = 9090
|
# metrics_port = 9090
|
||||||
# metrics_listen = "0.0.0.0:9090" # Адрес прослушивания для метрик (переопределяет metrics_port)
|
# metrics_listen = "127.0.0.1:9090" # Адрес прослушивания для метрик (переопределяет metrics_port)
|
||||||
# metrics_whitelist = ["127.0.0.1", "::1", "0.0.0.0/0"]
|
# metrics_whitelist = ["127.0.0.1/32", "::1/128"]
|
||||||
|
|
||||||
[server.api]
|
[server.api]
|
||||||
enabled = true
|
enabled = true
|
||||||
listen = "0.0.0.0:9091"
|
listen = "127.0.0.1:9091"
|
||||||
whitelist = ["127.0.0.0/8"]
|
whitelist = ["127.0.0.1/32", "::1/128"]
|
||||||
minimal_runtime_enabled = false
|
minimal_runtime_enabled = false
|
||||||
minimal_runtime_cache_ttl_ms = 1000
|
minimal_runtime_cache_ttl_ms = 1000
|
||||||
|
|
||||||
@@ -128,9 +154,9 @@ ip = "0.0.0.0"
|
|||||||
|
|
||||||
# === Обход блокировок и маскировка ===
|
# === Обход блокировок и маскировка ===
|
||||||
[censorship]
|
[censorship]
|
||||||
tls_domain = "petrovich.ru"
|
tls_domain = "petrovich.ru" # Домен Fake-TLS / SNI, который будет использоваться в сгенерированных ee-ссылках
|
||||||
mask = true
|
mask = true
|
||||||
tls_emulation = true # Получить реальную длину сертификата и эмулировать запись TLS
|
tls_emulation = true # Получить реальную длину сертификата и эмулировать запись TLS
|
||||||
tls_front_dir = "tlsfront" # Директория кэша для эмуляции TLS
|
tls_front_dir = "tlsfront" # Директория кэша для эмуляции TLS
|
||||||
|
|
||||||
[access.users]
|
[access.users]
|
||||||
@@ -141,9 +167,9 @@ hello = "00000000000000000000000000000000"
|
|||||||
Затем нажмите Ctrl+S -> Ctrl+X, чтобы сохранить
|
Затем нажмите Ctrl+S -> Ctrl+X, чтобы сохранить
|
||||||
|
|
||||||
> [!WARNING]
|
> [!WARNING]
|
||||||
> Замените значение параметра hello на значение, которое вы получили в пункте 0.
|
> Замените значение параметра `hello` на значение, которое вы получили в пункте 0.
|
||||||
> Так же замените значение параметра tls_domain на другой сайт.
|
> Так же замените значение параметра `tls_domain` на другой сайт.
|
||||||
> Изменение параметра tls_domain сделает нерабочими все ссылки, использующие старый домен!
|
> Изменение параметра `tls_domain` сделает нерабочими все ссылки, использующие старый домен!
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|||||||
+38
-14
@@ -1,6 +1,6 @@
|
|||||||
#![allow(clippy::too_many_arguments)]
|
#![allow(clippy::too_many_arguments)]
|
||||||
|
|
||||||
use std::convert::Infallible;
|
use std::io::{Error as IoError, ErrorKind};
|
||||||
use std::net::{IpAddr, SocketAddr};
|
use std::net::{IpAddr, SocketAddr};
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
@@ -16,7 +16,7 @@ use tokio::net::TcpListener;
|
|||||||
use tokio::sync::{Mutex, RwLock, watch};
|
use tokio::sync::{Mutex, RwLock, watch};
|
||||||
use tracing::{debug, info, warn};
|
use tracing::{debug, info, warn};
|
||||||
|
|
||||||
use crate::config::ProxyConfig;
|
use crate::config::{ApiGrayAction, ProxyConfig};
|
||||||
use crate::ip_tracker::UserIpTracker;
|
use crate::ip_tracker::UserIpTracker;
|
||||||
use crate::proxy::route_mode::RouteRuntimeController;
|
use crate::proxy::route_mode::RouteRuntimeController;
|
||||||
use crate::startup::StartupTracker;
|
use crate::startup::StartupTracker;
|
||||||
@@ -184,7 +184,9 @@ pub async fn serve(
|
|||||||
.serve_connection(hyper_util::rt::TokioIo::new(stream), svc)
|
.serve_connection(hyper_util::rt::TokioIo::new(stream), svc)
|
||||||
.await
|
.await
|
||||||
{
|
{
|
||||||
debug!(error = %error, "API connection error");
|
if !error.is_user() {
|
||||||
|
debug!(error = %error, "API connection error");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -195,7 +197,7 @@ async fn handle(
|
|||||||
peer: SocketAddr,
|
peer: SocketAddr,
|
||||||
shared: Arc<ApiShared>,
|
shared: Arc<ApiShared>,
|
||||||
config_rx: watch::Receiver<Arc<ProxyConfig>>,
|
config_rx: watch::Receiver<Arc<ProxyConfig>>,
|
||||||
) -> Result<Response<Full<Bytes>>, Infallible> {
|
) -> Result<Response<Full<Bytes>>, IoError> {
|
||||||
let request_id = shared.next_request_id();
|
let request_id = shared.next_request_id();
|
||||||
let cfg = config_rx.borrow().clone();
|
let cfg = config_rx.borrow().clone();
|
||||||
let api_cfg = &cfg.server.api;
|
let api_cfg = &cfg.server.api;
|
||||||
@@ -213,14 +215,25 @@ async fn handle(
|
|||||||
|
|
||||||
if !api_cfg.whitelist.is_empty() && !api_cfg.whitelist.iter().any(|net| net.contains(peer.ip()))
|
if !api_cfg.whitelist.is_empty() && !api_cfg.whitelist.iter().any(|net| net.contains(peer.ip()))
|
||||||
{
|
{
|
||||||
return Ok(error_response(
|
return match api_cfg.gray_action {
|
||||||
request_id,
|
ApiGrayAction::Api => Ok(error_response(
|
||||||
ApiFailure::new(
|
request_id,
|
||||||
StatusCode::FORBIDDEN,
|
ApiFailure::new(
|
||||||
"forbidden",
|
StatusCode::FORBIDDEN,
|
||||||
"Source IP is not allowed",
|
"forbidden",
|
||||||
),
|
"Source IP is not allowed",
|
||||||
));
|
),
|
||||||
|
)),
|
||||||
|
ApiGrayAction::Ok200 => Ok(Response::builder()
|
||||||
|
.status(StatusCode::OK)
|
||||||
|
.header("content-type", "text/html; charset=utf-8")
|
||||||
|
.body(Full::new(Bytes::new()))
|
||||||
|
.unwrap()),
|
||||||
|
ApiGrayAction::Drop => Err(IoError::new(
|
||||||
|
ErrorKind::ConnectionAborted,
|
||||||
|
"api request dropped by gray_action=drop",
|
||||||
|
)),
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if !api_cfg.auth_header.is_empty() {
|
if !api_cfg.auth_header.is_empty() {
|
||||||
@@ -244,11 +257,16 @@ async fn handle(
|
|||||||
|
|
||||||
let method = req.method().clone();
|
let method = req.method().clone();
|
||||||
let path = req.uri().path().to_string();
|
let path = req.uri().path().to_string();
|
||||||
|
let normalized_path = if path.len() > 1 {
|
||||||
|
path.trim_end_matches('/')
|
||||||
|
} else {
|
||||||
|
path.as_str()
|
||||||
|
};
|
||||||
let query = req.uri().query().map(str::to_string);
|
let query = req.uri().query().map(str::to_string);
|
||||||
let body_limit = api_cfg.request_body_limit_bytes;
|
let body_limit = api_cfg.request_body_limit_bytes;
|
||||||
|
|
||||||
let result: Result<Response<Full<Bytes>>, ApiFailure> = async {
|
let result: Result<Response<Full<Bytes>>, ApiFailure> = async {
|
||||||
match (method.as_str(), path.as_str()) {
|
match (method.as_str(), normalized_path) {
|
||||||
("GET", "/v1/health") => {
|
("GET", "/v1/health") => {
|
||||||
let revision = current_revision(&shared.config_path).await?;
|
let revision = current_revision(&shared.config_path).await?;
|
||||||
let data = HealthData {
|
let data = HealthData {
|
||||||
@@ -431,7 +449,7 @@ async fn handle(
|
|||||||
Ok(success_response(status, data, revision))
|
Ok(success_response(status, data, revision))
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
if let Some(user) = path.strip_prefix("/v1/users/")
|
if let Some(user) = normalized_path.strip_prefix("/v1/users/")
|
||||||
&& !user.is_empty()
|
&& !user.is_empty()
|
||||||
&& !user.contains('/')
|
&& !user.contains('/')
|
||||||
{
|
{
|
||||||
@@ -600,6 +618,12 @@ async fn handle(
|
|||||||
),
|
),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
debug!(
|
||||||
|
method = method.as_str(),
|
||||||
|
path = %path,
|
||||||
|
normalized_path = %normalized_path,
|
||||||
|
"API route not found"
|
||||||
|
);
|
||||||
Ok(error_response(
|
Ok(error_response(
|
||||||
request_id,
|
request_id,
|
||||||
ApiFailure::new(StatusCode::NOT_FOUND, "not_found", "Route not found"),
|
ApiFailure::new(StatusCode::NOT_FOUND, "not_found", "Route not found"),
|
||||||
|
|||||||
+13
-1
@@ -452,7 +452,11 @@ fn build_user_links(
|
|||||||
startup_detected_ip_v6: Option<IpAddr>,
|
startup_detected_ip_v6: Option<IpAddr>,
|
||||||
) -> UserLinks {
|
) -> UserLinks {
|
||||||
let hosts = resolve_link_hosts(cfg, startup_detected_ip_v4, startup_detected_ip_v6);
|
let hosts = resolve_link_hosts(cfg, startup_detected_ip_v4, startup_detected_ip_v6);
|
||||||
let port = cfg.general.links.public_port.unwrap_or(cfg.server.port);
|
let port = cfg
|
||||||
|
.general
|
||||||
|
.links
|
||||||
|
.public_port
|
||||||
|
.unwrap_or(resolve_default_link_port(cfg));
|
||||||
let tls_domains = resolve_tls_domains(cfg);
|
let tls_domains = resolve_tls_domains(cfg);
|
||||||
|
|
||||||
let mut classic = Vec::new();
|
let mut classic = Vec::new();
|
||||||
@@ -490,6 +494,14 @@ fn build_user_links(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn resolve_default_link_port(cfg: &ProxyConfig) -> u16 {
|
||||||
|
cfg.server
|
||||||
|
.listeners
|
||||||
|
.first()
|
||||||
|
.and_then(|listener| listener.port)
|
||||||
|
.unwrap_or(cfg.server.port)
|
||||||
|
}
|
||||||
|
|
||||||
fn resolve_link_hosts(
|
fn resolve_link_hosts(
|
||||||
cfg: &ProxyConfig,
|
cfg: &ProxyConfig,
|
||||||
startup_detected_ip_v4: Option<IpAddr>,
|
startup_detected_ip_v4: Option<IpAddr>,
|
||||||
|
|||||||
+2
-1
@@ -598,16 +598,17 @@ secure = false
|
|||||||
tls = true
|
tls = true
|
||||||
|
|
||||||
[server]
|
[server]
|
||||||
port = {port}
|
|
||||||
listen_addr_ipv4 = "0.0.0.0"
|
listen_addr_ipv4 = "0.0.0.0"
|
||||||
listen_addr_ipv6 = "::"
|
listen_addr_ipv6 = "::"
|
||||||
|
|
||||||
[[server.listeners]]
|
[[server.listeners]]
|
||||||
ip = "0.0.0.0"
|
ip = "0.0.0.0"
|
||||||
|
port = {port}
|
||||||
# reuse_allow = false # Set true only when intentionally running multiple telemt instances on same port
|
# reuse_allow = false # Set true only when intentionally running multiple telemt instances on same port
|
||||||
|
|
||||||
[[server.listeners]]
|
[[server.listeners]]
|
||||||
ip = "::"
|
ip = "::"
|
||||||
|
port = {port}
|
||||||
|
|
||||||
[timeouts]
|
[timeouts]
|
||||||
client_first_byte_idle_secs = 300
|
client_first_byte_idle_secs = 300
|
||||||
|
|||||||
@@ -17,8 +17,9 @@
|
|||||||
//! | `network` | `dns_overrides` | Applied immediately |
|
//! | `network` | `dns_overrides` | Applied immediately |
|
||||||
//! | `access` | All user/quota fields | Effective immediately |
|
//! | `access` | All user/quota fields | Effective immediately |
|
||||||
//!
|
//!
|
||||||
//! Fields that require re-binding sockets (`server.port`, `censorship.*`,
|
//! Fields that require re-binding sockets (`server.listeners`, legacy
|
||||||
//! `network.*`, `use_middle_proxy`) are **not** applied; a warning is emitted.
|
//! `server.port`, `censorship.*`, `network.*`, `use_middle_proxy`) are **not**
|
||||||
|
//! applied; a warning is emitted.
|
||||||
//! Non-hot changes are never mixed into the runtime config snapshot.
|
//! Non-hot changes are never mixed into the runtime config snapshot.
|
||||||
|
|
||||||
use std::collections::BTreeSet;
|
use std::collections::BTreeSet;
|
||||||
@@ -299,6 +300,7 @@ fn listeners_equal(
|
|||||||
}
|
}
|
||||||
lhs.iter().zip(rhs.iter()).all(|(a, b)| {
|
lhs.iter().zip(rhs.iter()).all(|(a, b)| {
|
||||||
a.ip == b.ip
|
a.ip == b.ip
|
||||||
|
&& a.port == b.port
|
||||||
&& a.announce == b.announce
|
&& a.announce == b.announce
|
||||||
&& a.announce_ip == b.announce_ip
|
&& a.announce_ip == b.announce_ip
|
||||||
&& a.proxy_protocol == b.proxy_protocol
|
&& a.proxy_protocol == b.proxy_protocol
|
||||||
@@ -306,6 +308,14 @@ fn listeners_equal(
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn resolve_default_link_port(cfg: &ProxyConfig) -> u16 {
|
||||||
|
cfg.server
|
||||||
|
.listeners
|
||||||
|
.first()
|
||||||
|
.and_then(|listener| listener.port)
|
||||||
|
.unwrap_or(cfg.server.port)
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default, PartialEq, Eq)]
|
#[derive(Debug, Clone, Default, PartialEq, Eq)]
|
||||||
struct WatchManifest {
|
struct WatchManifest {
|
||||||
files: BTreeSet<PathBuf>,
|
files: BTreeSet<PathBuf>,
|
||||||
@@ -560,6 +570,7 @@ fn warn_non_hot_changes(old: &ProxyConfig, new: &ProxyConfig, non_hot_changed: b
|
|||||||
if old.server.api.enabled != new.server.api.enabled
|
if old.server.api.enabled != new.server.api.enabled
|
||||||
|| old.server.api.listen != new.server.api.listen
|
|| old.server.api.listen != new.server.api.listen
|
||||||
|| old.server.api.whitelist != new.server.api.whitelist
|
|| old.server.api.whitelist != new.server.api.whitelist
|
||||||
|
|| old.server.api.gray_action != new.server.api.gray_action
|
||||||
|| old.server.api.auth_header != new.server.api.auth_header
|
|| old.server.api.auth_header != new.server.api.auth_header
|
||||||
|| old.server.api.request_body_limit_bytes != new.server.api.request_body_limit_bytes
|
|| old.server.api.request_body_limit_bytes != new.server.api.request_body_limit_bytes
|
||||||
|| old.server.api.minimal_runtime_enabled != new.server.api.minimal_runtime_enabled
|
|| old.server.api.minimal_runtime_enabled != new.server.api.minimal_runtime_enabled
|
||||||
@@ -1119,7 +1130,7 @@ fn log_changes(
|
|||||||
.general
|
.general
|
||||||
.links
|
.links
|
||||||
.public_port
|
.public_port
|
||||||
.unwrap_or(new_cfg.server.port);
|
.unwrap_or(resolve_default_link_port(new_cfg));
|
||||||
for user in &added {
|
for user in &added {
|
||||||
if let Some(secret) = new_hot.users.get(*user) {
|
if let Some(secret) = new_hot.users.get(*user) {
|
||||||
print_user_links(user, secret, &host, port, new_cfg);
|
print_user_links(user, secret, &host, port, new_cfg);
|
||||||
|
|||||||
@@ -253,6 +253,12 @@ fn validate_upstreams(config: &ProxyConfig) -> Result<()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for upstream in &config.upstreams {
|
for upstream in &config.upstreams {
|
||||||
|
if matches!(upstream.ipv4, Some(false)) && matches!(upstream.ipv6, Some(false)) {
|
||||||
|
return Err(ProxyError::Config(
|
||||||
|
"upstream.ipv4 and upstream.ipv6 cannot both be false".to_string(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
if let UpstreamType::Shadowsocks { url, .. } = &upstream.upstream_type {
|
if let UpstreamType::Shadowsocks { url, .. } = &upstream.upstream_type {
|
||||||
let parsed = ShadowsocksServerConfig::from_url(url)
|
let parsed = ShadowsocksServerConfig::from_url(url)
|
||||||
.map_err(|error| ProxyError::Config(format!("invalid shadowsocks url: {error}")))?;
|
.map_err(|error| ProxyError::Config(format!("invalid shadowsocks url: {error}")))?;
|
||||||
@@ -340,12 +346,29 @@ impl ProxyConfig {
|
|||||||
let update_every_is_explicit = general_table
|
let update_every_is_explicit = general_table
|
||||||
.map(|table| table.contains_key("update_every"))
|
.map(|table| table.contains_key("update_every"))
|
||||||
.unwrap_or(false);
|
.unwrap_or(false);
|
||||||
|
let beobachten_is_explicit = general_table
|
||||||
|
.map(|table| table.contains_key("beobachten"))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let beobachten_minutes_is_explicit = general_table
|
||||||
|
.map(|table| table.contains_key("beobachten_minutes"))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let beobachten_flush_secs_is_explicit = general_table
|
||||||
|
.map(|table| table.contains_key("beobachten_flush_secs"))
|
||||||
|
.unwrap_or(false);
|
||||||
|
let beobachten_file_is_explicit = general_table
|
||||||
|
.map(|table| table.contains_key("beobachten_file"))
|
||||||
|
.unwrap_or(false);
|
||||||
let legacy_secret_is_explicit = general_table
|
let legacy_secret_is_explicit = general_table
|
||||||
.map(|table| table.contains_key("proxy_secret_auto_reload_secs"))
|
.map(|table| table.contains_key("proxy_secret_auto_reload_secs"))
|
||||||
.unwrap_or(false);
|
.unwrap_or(false);
|
||||||
let legacy_config_is_explicit = general_table
|
let legacy_config_is_explicit = general_table
|
||||||
.map(|table| table.contains_key("proxy_config_auto_reload_secs"))
|
.map(|table| table.contains_key("proxy_config_auto_reload_secs"))
|
||||||
.unwrap_or(false);
|
.unwrap_or(false);
|
||||||
|
let legacy_top_level_beobachten = parsed_toml.get("beobachten").cloned();
|
||||||
|
let legacy_top_level_beobachten_minutes = parsed_toml.get("beobachten_minutes").cloned();
|
||||||
|
let legacy_top_level_beobachten_flush_secs =
|
||||||
|
parsed_toml.get("beobachten_flush_secs").cloned();
|
||||||
|
let legacy_top_level_beobachten_file = parsed_toml.get("beobachten_file").cloned();
|
||||||
let stun_servers_is_explicit = network_table
|
let stun_servers_is_explicit = network_table
|
||||||
.map(|table| table.contains_key("stun_servers"))
|
.map(|table| table.contains_key("stun_servers"))
|
||||||
.unwrap_or(false);
|
.unwrap_or(false);
|
||||||
@@ -358,6 +381,59 @@ impl ProxyConfig {
|
|||||||
config.general.update_every = None;
|
config.general.update_every = None;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Backward compatibility: legacy top-level beobachten* keys.
|
||||||
|
// Prefer `[general].*` when both are present.
|
||||||
|
let mut legacy_beobachten_applied = false;
|
||||||
|
if !beobachten_is_explicit && let Some(value) = legacy_top_level_beobachten.as_ref() {
|
||||||
|
let parsed = value.as_bool().ok_or_else(|| {
|
||||||
|
ProxyError::Config("beobachten (top-level) must be a boolean".to_string())
|
||||||
|
})?;
|
||||||
|
config.general.beobachten = parsed;
|
||||||
|
legacy_beobachten_applied = true;
|
||||||
|
}
|
||||||
|
if !beobachten_minutes_is_explicit
|
||||||
|
&& let Some(value) = legacy_top_level_beobachten_minutes.as_ref()
|
||||||
|
{
|
||||||
|
let raw = value.as_integer().ok_or_else(|| {
|
||||||
|
ProxyError::Config("beobachten_minutes (top-level) must be an integer".to_string())
|
||||||
|
})?;
|
||||||
|
let parsed = u64::try_from(raw).map_err(|_| {
|
||||||
|
ProxyError::Config(
|
||||||
|
"beobachten_minutes (top-level) must be within u64 range".to_string(),
|
||||||
|
)
|
||||||
|
})?;
|
||||||
|
config.general.beobachten_minutes = parsed;
|
||||||
|
legacy_beobachten_applied = true;
|
||||||
|
}
|
||||||
|
if !beobachten_flush_secs_is_explicit
|
||||||
|
&& let Some(value) = legacy_top_level_beobachten_flush_secs.as_ref()
|
||||||
|
{
|
||||||
|
let raw = value.as_integer().ok_or_else(|| {
|
||||||
|
ProxyError::Config(
|
||||||
|
"beobachten_flush_secs (top-level) must be an integer".to_string(),
|
||||||
|
)
|
||||||
|
})?;
|
||||||
|
let parsed = u64::try_from(raw).map_err(|_| {
|
||||||
|
ProxyError::Config(
|
||||||
|
"beobachten_flush_secs (top-level) must be within u64 range".to_string(),
|
||||||
|
)
|
||||||
|
})?;
|
||||||
|
config.general.beobachten_flush_secs = parsed;
|
||||||
|
legacy_beobachten_applied = true;
|
||||||
|
}
|
||||||
|
if !beobachten_file_is_explicit
|
||||||
|
&& let Some(value) = legacy_top_level_beobachten_file.as_ref()
|
||||||
|
{
|
||||||
|
let parsed = value.as_str().ok_or_else(|| {
|
||||||
|
ProxyError::Config("beobachten_file (top-level) must be a string".to_string())
|
||||||
|
})?;
|
||||||
|
config.general.beobachten_file = parsed.to_string();
|
||||||
|
legacy_beobachten_applied = true;
|
||||||
|
}
|
||||||
|
if legacy_beobachten_applied {
|
||||||
|
warn!("top-level beobachten* keys are deprecated; use general.beobachten* instead");
|
||||||
|
}
|
||||||
|
|
||||||
let legacy_nat_stun = config.general.middle_proxy_nat_stun.take();
|
let legacy_nat_stun = config.general.middle_proxy_nat_stun.take();
|
||||||
let legacy_nat_stun_servers =
|
let legacy_nat_stun_servers =
|
||||||
std::mem::take(&mut config.general.middle_proxy_nat_stun_servers);
|
std::mem::take(&mut config.general.middle_proxy_nat_stun_servers);
|
||||||
@@ -1250,6 +1326,7 @@ impl ProxyConfig {
|
|||||||
if let Ok(ipv4) = ipv4_str.parse::<IpAddr>() {
|
if let Ok(ipv4) = ipv4_str.parse::<IpAddr>() {
|
||||||
config.server.listeners.push(ListenerConfig {
|
config.server.listeners.push(ListenerConfig {
|
||||||
ip: ipv4,
|
ip: ipv4,
|
||||||
|
port: Some(config.server.port),
|
||||||
announce: None,
|
announce: None,
|
||||||
announce_ip: None,
|
announce_ip: None,
|
||||||
proxy_protocol: None,
|
proxy_protocol: None,
|
||||||
@@ -1261,6 +1338,7 @@ impl ProxyConfig {
|
|||||||
{
|
{
|
||||||
config.server.listeners.push(ListenerConfig {
|
config.server.listeners.push(ListenerConfig {
|
||||||
ip: ipv6,
|
ip: ipv6,
|
||||||
|
port: Some(config.server.port),
|
||||||
announce: None,
|
announce: None,
|
||||||
announce_ip: None,
|
announce_ip: None,
|
||||||
proxy_protocol: None,
|
proxy_protocol: None,
|
||||||
@@ -1269,6 +1347,13 @@ impl ProxyConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Migration: listeners[].port fallback to legacy server.port.
|
||||||
|
for listener in &mut config.server.listeners {
|
||||||
|
if listener.port.is_none() {
|
||||||
|
listener.port = Some(config.server.port);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Migration: announce_ip → announce for each listener.
|
// Migration: announce_ip → announce for each listener.
|
||||||
for listener in &mut config.server.listeners {
|
for listener in &mut config.server.listeners {
|
||||||
if listener.announce.is_none()
|
if listener.announce.is_none()
|
||||||
@@ -1289,11 +1374,14 @@ impl ProxyConfig {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1385,6 +1473,21 @@ mod tests {
|
|||||||
const TEST_SHADOWSOCKS_URL: &str =
|
const TEST_SHADOWSOCKS_URL: &str =
|
||||||
"ss://2022-blake3-aes-256-gcm:MDEyMzQ1Njc4OTAxMjM0NTY3ODkwMTIzNDU2Nzg5MDE=@127.0.0.1:8388";
|
"ss://2022-blake3-aes-256-gcm:MDEyMzQ1Njc4OTAxMjM0NTY3ODkwMTIzNDU2Nzg5MDE=@127.0.0.1:8388";
|
||||||
|
|
||||||
|
fn load_config_from_temp_toml(toml: &str) -> ProxyConfig {
|
||||||
|
let nonce = std::time::SystemTime::now()
|
||||||
|
.duration_since(std::time::UNIX_EPOCH)
|
||||||
|
.unwrap()
|
||||||
|
.as_nanos();
|
||||||
|
let dir = std::env::temp_dir().join(format!("telemt_load_cfg_{nonce}"));
|
||||||
|
std::fs::create_dir_all(&dir).unwrap();
|
||||||
|
let path = dir.join("config.toml");
|
||||||
|
std::fs::write(&path, toml).unwrap();
|
||||||
|
let cfg = ProxyConfig::load(&path).unwrap();
|
||||||
|
let _ = std::fs::remove_file(path);
|
||||||
|
let _ = std::fs::remove_dir(dir);
|
||||||
|
cfg
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn serde_defaults_remain_unchanged_for_present_sections() {
|
fn serde_defaults_remain_unchanged_for_present_sections() {
|
||||||
let toml = r#"
|
let toml = r#"
|
||||||
@@ -1481,6 +1584,7 @@ mod tests {
|
|||||||
cfg.general.rpc_proxy_req_every,
|
cfg.general.rpc_proxy_req_every,
|
||||||
default_rpc_proxy_req_every()
|
default_rpc_proxy_req_every()
|
||||||
);
|
);
|
||||||
|
assert_eq!(cfg.general.beobachten_file, default_beobachten_file());
|
||||||
assert_eq!(cfg.general.update_every, default_update_every());
|
assert_eq!(cfg.general.update_every, default_update_every());
|
||||||
assert_eq!(cfg.server.listen_addr_ipv4, default_listen_addr_ipv4());
|
assert_eq!(cfg.server.listen_addr_ipv4, default_listen_addr_ipv4());
|
||||||
assert_eq!(cfg.server.listen_addr_ipv6, default_listen_addr_ipv6_opt());
|
assert_eq!(cfg.server.listen_addr_ipv6, default_listen_addr_ipv6_opt());
|
||||||
@@ -1491,6 +1595,7 @@ mod tests {
|
|||||||
assert_eq!(cfg.censorship.unknown_sni_action, UnknownSniAction::Drop);
|
assert_eq!(cfg.censorship.unknown_sni_action, UnknownSniAction::Drop);
|
||||||
assert_eq!(cfg.server.api.listen, default_api_listen());
|
assert_eq!(cfg.server.api.listen, default_api_listen());
|
||||||
assert_eq!(cfg.server.api.whitelist, default_api_whitelist());
|
assert_eq!(cfg.server.api.whitelist, default_api_whitelist());
|
||||||
|
assert_eq!(cfg.server.api.gray_action, ApiGrayAction::Drop);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
cfg.server.api.request_body_limit_bytes,
|
cfg.server.api.request_body_limit_bytes,
|
||||||
default_api_request_body_limit_bytes()
|
default_api_request_body_limit_bytes()
|
||||||
@@ -1647,6 +1752,7 @@ mod tests {
|
|||||||
default_upstream_connect_failfast_hard_errors()
|
default_upstream_connect_failfast_hard_errors()
|
||||||
);
|
);
|
||||||
assert_eq!(general.rpc_proxy_req_every, default_rpc_proxy_req_every());
|
assert_eq!(general.rpc_proxy_req_every, default_rpc_proxy_req_every());
|
||||||
|
assert_eq!(general.beobachten_file, default_beobachten_file());
|
||||||
assert_eq!(general.update_every, default_update_every());
|
assert_eq!(general.update_every, default_update_every());
|
||||||
|
|
||||||
let server = ServerConfig::default();
|
let server = ServerConfig::default();
|
||||||
@@ -1661,6 +1767,7 @@ mod tests {
|
|||||||
);
|
);
|
||||||
assert_eq!(server.api.listen, default_api_listen());
|
assert_eq!(server.api.listen, default_api_listen());
|
||||||
assert_eq!(server.api.whitelist, default_api_whitelist());
|
assert_eq!(server.api.whitelist, default_api_whitelist());
|
||||||
|
assert_eq!(server.api.gray_action, ApiGrayAction::Drop);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
server.api.request_body_limit_bytes,
|
server.api.request_body_limit_bytes,
|
||||||
default_api_request_body_limit_bytes()
|
default_api_request_body_limit_bytes()
|
||||||
@@ -1808,6 +1915,107 @@ mod tests {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn api_gray_action_parses_and_defaults_to_drop() {
|
||||||
|
let cfg_default: ProxyConfig = toml::from_str(
|
||||||
|
r#"
|
||||||
|
[server]
|
||||||
|
[general]
|
||||||
|
[network]
|
||||||
|
[access]
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(cfg_default.server.api.gray_action, ApiGrayAction::Drop);
|
||||||
|
|
||||||
|
let cfg_api: ProxyConfig = toml::from_str(
|
||||||
|
r#"
|
||||||
|
[server]
|
||||||
|
[general]
|
||||||
|
[network]
|
||||||
|
[access]
|
||||||
|
[server.api]
|
||||||
|
gray_action = "api"
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(cfg_api.server.api.gray_action, ApiGrayAction::Api);
|
||||||
|
|
||||||
|
let cfg_200: ProxyConfig = toml::from_str(
|
||||||
|
r#"
|
||||||
|
[server]
|
||||||
|
[general]
|
||||||
|
[network]
|
||||||
|
[access]
|
||||||
|
[server.api]
|
||||||
|
gray_action = "200"
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(cfg_200.server.api.gray_action, ApiGrayAction::Ok200);
|
||||||
|
|
||||||
|
let cfg_drop: ProxyConfig = toml::from_str(
|
||||||
|
r#"
|
||||||
|
[server]
|
||||||
|
[general]
|
||||||
|
[network]
|
||||||
|
[access]
|
||||||
|
[server.api]
|
||||||
|
gray_action = "drop"
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(cfg_drop.server.api.gray_action, ApiGrayAction::Drop);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn top_level_beobachten_keys_migrate_to_general_when_general_not_explicit() {
|
||||||
|
let cfg = load_config_from_temp_toml(
|
||||||
|
r#"
|
||||||
|
beobachten = false
|
||||||
|
beobachten_minutes = 7
|
||||||
|
beobachten_flush_secs = 3
|
||||||
|
beobachten_file = "tmp/legacy-beob.txt"
|
||||||
|
|
||||||
|
[server]
|
||||||
|
[general]
|
||||||
|
[network]
|
||||||
|
[access]
|
||||||
|
"#,
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(!cfg.general.beobachten);
|
||||||
|
assert_eq!(cfg.general.beobachten_minutes, 7);
|
||||||
|
assert_eq!(cfg.general.beobachten_flush_secs, 3);
|
||||||
|
assert_eq!(cfg.general.beobachten_file, "tmp/legacy-beob.txt");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn general_beobachten_keys_have_priority_over_legacy_top_level() {
|
||||||
|
let cfg = load_config_from_temp_toml(
|
||||||
|
r#"
|
||||||
|
beobachten = true
|
||||||
|
beobachten_minutes = 30
|
||||||
|
beobachten_flush_secs = 30
|
||||||
|
beobachten_file = "tmp/legacy-beob.txt"
|
||||||
|
|
||||||
|
[server]
|
||||||
|
[general]
|
||||||
|
beobachten = false
|
||||||
|
beobachten_minutes = 5
|
||||||
|
beobachten_flush_secs = 2
|
||||||
|
beobachten_file = "tmp/general-beob.txt"
|
||||||
|
[network]
|
||||||
|
[access]
|
||||||
|
"#,
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(!cfg.general.beobachten);
|
||||||
|
assert_eq!(cfg.general.beobachten_minutes, 5);
|
||||||
|
assert_eq!(cfg.general.beobachten_flush_secs, 2);
|
||||||
|
assert_eq!(cfg.general.beobachten_file, "tmp/general-beob.txt");
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn dc_overrides_allow_string_and_array() {
|
fn dc_overrides_allow_string_and_array() {
|
||||||
let toml = r#"
|
let toml = r#"
|
||||||
|
|||||||
+40
-1
@@ -1153,7 +1153,8 @@ pub struct LinksConfig {
|
|||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub public_host: Option<String>,
|
pub public_host: Option<String>,
|
||||||
|
|
||||||
/// Public port for tg:// link generation (overrides server.port).
|
/// Public port for tg:// link generation.
|
||||||
|
/// Overrides listener ports and legacy `server.port`.
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub public_port: Option<u16>,
|
pub public_port: Option<u16>,
|
||||||
}
|
}
|
||||||
@@ -1183,6 +1184,13 @@ pub struct ApiConfig {
|
|||||||
#[serde(default = "default_api_whitelist")]
|
#[serde(default = "default_api_whitelist")]
|
||||||
pub whitelist: Vec<IpNetwork>,
|
pub whitelist: Vec<IpNetwork>,
|
||||||
|
|
||||||
|
/// Behavior for requests from source IPs outside `whitelist`.
|
||||||
|
/// - `api`: return structured API forbidden response.
|
||||||
|
/// - `200`: return `200 OK` with an empty body.
|
||||||
|
/// - `drop`: close the connection without HTTP response.
|
||||||
|
#[serde(default)]
|
||||||
|
pub gray_action: ApiGrayAction,
|
||||||
|
|
||||||
/// Optional static value for `Authorization` header validation.
|
/// Optional static value for `Authorization` header validation.
|
||||||
/// Empty string disables header auth.
|
/// Empty string disables header auth.
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
@@ -1227,6 +1235,7 @@ impl Default for ApiConfig {
|
|||||||
enabled: default_true(),
|
enabled: default_true(),
|
||||||
listen: default_api_listen(),
|
listen: default_api_listen(),
|
||||||
whitelist: default_api_whitelist(),
|
whitelist: default_api_whitelist(),
|
||||||
|
gray_action: ApiGrayAction::default(),
|
||||||
auth_header: String::new(),
|
auth_header: String::new(),
|
||||||
request_body_limit_bytes: default_api_request_body_limit_bytes(),
|
request_body_limit_bytes: default_api_request_body_limit_bytes(),
|
||||||
minimal_runtime_enabled: default_api_minimal_runtime_enabled(),
|
minimal_runtime_enabled: default_api_minimal_runtime_enabled(),
|
||||||
@@ -1240,6 +1249,19 @@ impl Default for ApiConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize, Default)]
|
||||||
|
#[serde(rename_all = "lowercase")]
|
||||||
|
pub enum ApiGrayAction {
|
||||||
|
/// Preserve current API behavior for denied source IPs.
|
||||||
|
Api,
|
||||||
|
/// Mimic a plain web endpoint by returning `200 OK` with an empty body.
|
||||||
|
#[serde(rename = "200")]
|
||||||
|
Ok200,
|
||||||
|
/// Drop connection without HTTP response for denied source IPs.
|
||||||
|
#[default]
|
||||||
|
Drop,
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize, Default)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize, Default)]
|
||||||
#[serde(rename_all = "lowercase")]
|
#[serde(rename_all = "lowercase")]
|
||||||
pub enum ConntrackMode {
|
pub enum ConntrackMode {
|
||||||
@@ -1354,6 +1376,8 @@ impl Default for ConntrackControlConfig {
|
|||||||
|
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
pub struct ServerConfig {
|
pub struct ServerConfig {
|
||||||
|
/// Legacy listener port used for backward compatibility.
|
||||||
|
/// For new configs prefer `[[server.listeners]].port`.
|
||||||
#[serde(default = "default_port")]
|
#[serde(default = "default_port")]
|
||||||
pub port: u16,
|
pub port: u16,
|
||||||
|
|
||||||
@@ -1856,6 +1880,10 @@ pub enum UpstreamType {
|
|||||||
interface: Option<String>,
|
interface: Option<String>,
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
bind_addresses: Option<Vec<String>>,
|
bind_addresses: Option<Vec<String>>,
|
||||||
|
/// Linux-only hard interface pinning via `SO_BINDTODEVICE`.
|
||||||
|
/// Optional alias: `force_bind`.
|
||||||
|
#[serde(default, alias = "force_bind")]
|
||||||
|
bindtodevice: Option<String>,
|
||||||
},
|
},
|
||||||
Socks4 {
|
Socks4 {
|
||||||
address: String,
|
address: String,
|
||||||
@@ -1892,11 +1920,22 @@ pub struct UpstreamConfig {
|
|||||||
pub scopes: String,
|
pub scopes: String,
|
||||||
#[serde(skip)]
|
#[serde(skip)]
|
||||||
pub selected_scope: String,
|
pub selected_scope: String,
|
||||||
|
/// Allow IPv4 DC targets for this upstream.
|
||||||
|
/// `None` means auto-detect from runtime connectivity state.
|
||||||
|
#[serde(default)]
|
||||||
|
pub ipv4: Option<bool>,
|
||||||
|
/// Allow IPv6 DC targets for this upstream.
|
||||||
|
/// `None` means auto-detect from runtime connectivity state.
|
||||||
|
#[serde(default)]
|
||||||
|
pub ipv6: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
pub struct ListenerConfig {
|
pub struct ListenerConfig {
|
||||||
pub ip: IpAddr,
|
pub ip: IpAddr,
|
||||||
|
/// Per-listener TCP port. If omitted, falls back to legacy `server.port`.
|
||||||
|
#[serde(default)]
|
||||||
|
pub port: Option<u16>,
|
||||||
/// IP address or hostname to announce in proxy links.
|
/// IP address or hostname to announce in proxy links.
|
||||||
/// Takes precedence over `announce_ip` if both are set.
|
/// Takes precedence over `announce_ip` if both are set.
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
|
|||||||
+40
-21
@@ -343,15 +343,28 @@ fn command_exists(binary: &str) -> bool {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn notrack_targets(cfg: &ProxyConfig) -> (Vec<Option<IpAddr>>, Vec<Option<IpAddr>>) {
|
fn listener_port_set(cfg: &ProxyConfig) -> Vec<u16> {
|
||||||
|
let mut ports: BTreeSet<u16> = BTreeSet::new();
|
||||||
|
if cfg.server.listeners.is_empty() {
|
||||||
|
ports.insert(cfg.server.port);
|
||||||
|
} else {
|
||||||
|
for listener in &cfg.server.listeners {
|
||||||
|
ports.insert(listener.port.unwrap_or(cfg.server.port));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ports.into_iter().collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn notrack_targets(cfg: &ProxyConfig) -> (Vec<(Option<IpAddr>, u16)>, Vec<(Option<IpAddr>, u16)>) {
|
||||||
let mode = cfg.server.conntrack_control.mode;
|
let mode = cfg.server.conntrack_control.mode;
|
||||||
let mut v4_targets: BTreeSet<Option<IpAddr>> = BTreeSet::new();
|
let mut v4_targets: BTreeSet<(Option<IpAddr>, u16)> = BTreeSet::new();
|
||||||
let mut v6_targets: BTreeSet<Option<IpAddr>> = BTreeSet::new();
|
let mut v6_targets: BTreeSet<(Option<IpAddr>, u16)> = BTreeSet::new();
|
||||||
|
|
||||||
match mode {
|
match mode {
|
||||||
ConntrackMode::Tracked => {}
|
ConntrackMode::Tracked => {}
|
||||||
ConntrackMode::Notrack => {
|
ConntrackMode::Notrack => {
|
||||||
if cfg.server.listeners.is_empty() {
|
if cfg.server.listeners.is_empty() {
|
||||||
|
let port = cfg.server.port;
|
||||||
if let Some(ipv4) = cfg
|
if let Some(ipv4) = cfg
|
||||||
.server
|
.server
|
||||||
.listen_addr_ipv4
|
.listen_addr_ipv4
|
||||||
@@ -359,9 +372,9 @@ fn notrack_targets(cfg: &ProxyConfig) -> (Vec<Option<IpAddr>>, Vec<Option<IpAddr
|
|||||||
.and_then(|s| s.parse::<IpAddr>().ok())
|
.and_then(|s| s.parse::<IpAddr>().ok())
|
||||||
{
|
{
|
||||||
if ipv4.is_unspecified() {
|
if ipv4.is_unspecified() {
|
||||||
v4_targets.insert(None);
|
v4_targets.insert((None, port));
|
||||||
} else {
|
} else {
|
||||||
v4_targets.insert(Some(ipv4));
|
v4_targets.insert((Some(ipv4), port));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if let Some(ipv6) = cfg
|
if let Some(ipv6) = cfg
|
||||||
@@ -371,33 +384,39 @@ fn notrack_targets(cfg: &ProxyConfig) -> (Vec<Option<IpAddr>>, Vec<Option<IpAddr
|
|||||||
.and_then(|s| s.parse::<IpAddr>().ok())
|
.and_then(|s| s.parse::<IpAddr>().ok())
|
||||||
{
|
{
|
||||||
if ipv6.is_unspecified() {
|
if ipv6.is_unspecified() {
|
||||||
v6_targets.insert(None);
|
v6_targets.insert((None, port));
|
||||||
} else {
|
} else {
|
||||||
v6_targets.insert(Some(ipv6));
|
v6_targets.insert((Some(ipv6), port));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
for listener in &cfg.server.listeners {
|
for listener in &cfg.server.listeners {
|
||||||
|
let port = listener.port.unwrap_or(cfg.server.port);
|
||||||
if listener.ip.is_ipv4() {
|
if listener.ip.is_ipv4() {
|
||||||
if listener.ip.is_unspecified() {
|
if listener.ip.is_unspecified() {
|
||||||
v4_targets.insert(None);
|
v4_targets.insert((None, port));
|
||||||
} else {
|
} else {
|
||||||
v4_targets.insert(Some(listener.ip));
|
v4_targets.insert((Some(listener.ip), port));
|
||||||
}
|
}
|
||||||
} else if listener.ip.is_unspecified() {
|
} else if listener.ip.is_unspecified() {
|
||||||
v6_targets.insert(None);
|
v6_targets.insert((None, port));
|
||||||
} else {
|
} else {
|
||||||
v6_targets.insert(Some(listener.ip));
|
v6_targets.insert((Some(listener.ip), port));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ConntrackMode::Hybrid => {
|
ConntrackMode::Hybrid => {
|
||||||
|
let ports = listener_port_set(cfg);
|
||||||
for ip in &cfg.server.conntrack_control.hybrid_listener_ips {
|
for ip in &cfg.server.conntrack_control.hybrid_listener_ips {
|
||||||
if ip.is_ipv4() {
|
if ip.is_ipv4() {
|
||||||
v4_targets.insert(Some(*ip));
|
for port in &ports {
|
||||||
|
v4_targets.insert((Some(*ip), *port));
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
v6_targets.insert(Some(*ip));
|
for port in &ports {
|
||||||
|
v6_targets.insert((Some(*ip), *port));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -422,19 +441,19 @@ async fn apply_nft_rules(cfg: &ProxyConfig) -> Result<(), String> {
|
|||||||
|
|
||||||
let (v4_targets, v6_targets) = notrack_targets(cfg);
|
let (v4_targets, v6_targets) = notrack_targets(cfg);
|
||||||
let mut rules = Vec::new();
|
let mut rules = Vec::new();
|
||||||
for ip in v4_targets {
|
for (ip, port) in v4_targets {
|
||||||
let rule = if let Some(ip) = ip {
|
let rule = if let Some(ip) = ip {
|
||||||
format!("tcp dport {} ip daddr {} notrack", cfg.server.port, ip)
|
format!("tcp dport {} ip daddr {} notrack", port, ip)
|
||||||
} else {
|
} else {
|
||||||
format!("tcp dport {} notrack", cfg.server.port)
|
format!("tcp dport {} notrack", port)
|
||||||
};
|
};
|
||||||
rules.push(rule);
|
rules.push(rule);
|
||||||
}
|
}
|
||||||
for ip in v6_targets {
|
for (ip, port) in v6_targets {
|
||||||
let rule = if let Some(ip) = ip {
|
let rule = if let Some(ip) = ip {
|
||||||
format!("tcp dport {} ip6 daddr {} notrack", cfg.server.port, ip)
|
format!("tcp dport {} ip6 daddr {} notrack", port, ip)
|
||||||
} else {
|
} else {
|
||||||
format!("tcp dport {} notrack", cfg.server.port)
|
format!("tcp dport {} notrack", port)
|
||||||
};
|
};
|
||||||
rules.push(rule);
|
rules.push(rule);
|
||||||
}
|
}
|
||||||
@@ -498,7 +517,7 @@ async fn apply_iptables_rules_for_binary(
|
|||||||
|
|
||||||
let (v4_targets, v6_targets) = notrack_targets(cfg);
|
let (v4_targets, v6_targets) = notrack_targets(cfg);
|
||||||
let selected = if ipv4 { v4_targets } else { v6_targets };
|
let selected = if ipv4 { v4_targets } else { v6_targets };
|
||||||
for ip in selected {
|
for (ip, port) in selected {
|
||||||
let mut args = vec![
|
let mut args = vec![
|
||||||
"-t".to_string(),
|
"-t".to_string(),
|
||||||
"raw".to_string(),
|
"raw".to_string(),
|
||||||
@@ -507,7 +526,7 @@ async fn apply_iptables_rules_for_binary(
|
|||||||
"-p".to_string(),
|
"-p".to_string(),
|
||||||
"tcp".to_string(),
|
"tcp".to_string(),
|
||||||
"--dport".to_string(),
|
"--dport".to_string(),
|
||||||
cfg.server.port.to_string(),
|
port.to_string(),
|
||||||
];
|
];
|
||||||
if let Some(ip) = ip {
|
if let Some(ip) = ip {
|
||||||
args.push("-d".to_string());
|
args.push("-d".to_string());
|
||||||
|
|||||||
@@ -31,6 +31,19 @@ pub(crate) struct BoundListeners {
|
|||||||
pub(crate) has_unix_listener: bool,
|
pub(crate) has_unix_listener: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn listener_port_or_legacy(listener: &crate::config::ListenerConfig, config: &ProxyConfig) -> u16 {
|
||||||
|
listener.port.unwrap_or(config.server.port)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn default_link_port(config: &ProxyConfig) -> u16 {
|
||||||
|
config
|
||||||
|
.server
|
||||||
|
.listeners
|
||||||
|
.first()
|
||||||
|
.and_then(|listener| listener.port)
|
||||||
|
.unwrap_or(config.server.port)
|
||||||
|
}
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
pub(crate) async fn bind_listeners(
|
pub(crate) async fn bind_listeners(
|
||||||
config: &Arc<ProxyConfig>,
|
config: &Arc<ProxyConfig>,
|
||||||
@@ -63,7 +76,8 @@ pub(crate) async fn bind_listeners(
|
|||||||
let mut listeners = Vec::new();
|
let mut listeners = Vec::new();
|
||||||
|
|
||||||
for listener_conf in &config.server.listeners {
|
for listener_conf in &config.server.listeners {
|
||||||
let addr = SocketAddr::new(listener_conf.ip, config.server.port);
|
let listener_port = listener_port_or_legacy(listener_conf, config);
|
||||||
|
let addr = SocketAddr::new(listener_conf.ip, listener_port);
|
||||||
if addr.is_ipv4() && !decision_ipv4_dc {
|
if addr.is_ipv4() && !decision_ipv4_dc {
|
||||||
warn!(%addr, "Skipping IPv4 listener: IPv4 disabled by [network]");
|
warn!(%addr, "Skipping IPv4 listener: IPv4 disabled by [network]");
|
||||||
continue;
|
continue;
|
||||||
@@ -106,11 +120,7 @@ pub(crate) async fn bind_listeners(
|
|||||||
if config.general.links.public_host.is_none()
|
if config.general.links.public_host.is_none()
|
||||||
&& !config.general.links.show.is_empty()
|
&& !config.general.links.show.is_empty()
|
||||||
{
|
{
|
||||||
let link_port = config
|
let link_port = config.general.links.public_port.unwrap_or(listener_port);
|
||||||
.general
|
|
||||||
.links
|
|
||||||
.public_port
|
|
||||||
.unwrap_or(config.server.port);
|
|
||||||
print_proxy_links(&public_host, link_port, config);
|
print_proxy_links(&public_host, link_port, config);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -158,7 +168,7 @@ pub(crate) async fn bind_listeners(
|
|||||||
.general
|
.general
|
||||||
.links
|
.links
|
||||||
.public_port
|
.public_port
|
||||||
.unwrap_or(config.server.port),
|
.unwrap_or(default_link_port(config)),
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
let ip = detected_ip_v4.or(detected_ip_v6).map(|ip| ip.to_string());
|
let ip = detected_ip_v4.or(detected_ip_v6).map(|ip| ip.to_string());
|
||||||
@@ -173,7 +183,7 @@ pub(crate) async fn bind_listeners(
|
|||||||
.general
|
.general
|
||||||
.links
|
.links
|
||||||
.public_port
|
.public_port
|
||||||
.unwrap_or(config.server.port),
|
.unwrap_or(default_link_port(config)),
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
+42
-27
@@ -81,23 +81,11 @@ pub async fn run() -> std::result::Result<(), Box<dyn std::error::Error>> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(unix)]
|
// Shared maestro startup and main loop. `drop_after_bind` runs on Unix after listeners are bound
|
||||||
async fn run_inner(
|
// (for privilege drop); it is a no-op on other platforms.
|
||||||
daemon_opts: DaemonOptions,
|
async fn run_telemt_core(
|
||||||
|
drop_after_bind: impl FnOnce(),
|
||||||
) -> std::result::Result<(), Box<dyn std::error::Error>> {
|
) -> std::result::Result<(), Box<dyn std::error::Error>> {
|
||||||
// Acquire PID file if daemonizing or if explicitly requested
|
|
||||||
// Keep it alive until shutdown (underscore prefix = intentionally kept for RAII cleanup)
|
|
||||||
let _pid_file = if daemon_opts.daemonize || daemon_opts.pid_file.is_some() {
|
|
||||||
let mut pf = PidFile::new(daemon_opts.pid_file_path());
|
|
||||||
if let Err(e) = pf.acquire() {
|
|
||||||
eprintln!("[telemt] {}", e);
|
|
||||||
std::process::exit(1);
|
|
||||||
}
|
|
||||||
Some(pf)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
};
|
|
||||||
|
|
||||||
let process_started_at = Instant::now();
|
let process_started_at = Instant::now();
|
||||||
let process_started_at_epoch_secs = SystemTime::now()
|
let process_started_at_epoch_secs = SystemTime::now()
|
||||||
.duration_since(UNIX_EPOCH)
|
.duration_since(UNIX_EPOCH)
|
||||||
@@ -761,17 +749,8 @@ async fn run_inner(
|
|||||||
std::process::exit(1);
|
std::process::exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Drop privileges after binding sockets (which may require root for port < 1024)
|
// On Unix, caller supplies privilege drop after bind (may require root for port < 1024).
|
||||||
if daemon_opts.user.is_some() || daemon_opts.group.is_some() {
|
drop_after_bind();
|
||||||
if let Err(e) = drop_privileges(
|
|
||||||
daemon_opts.user.as_deref(),
|
|
||||||
daemon_opts.group.as_deref(),
|
|
||||||
_pid_file.as_ref(),
|
|
||||||
) {
|
|
||||||
error!(error = %e, "Failed to drop privileges");
|
|
||||||
std::process::exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
runtime_tasks::apply_runtime_log_filter(
|
runtime_tasks::apply_runtime_log_filter(
|
||||||
has_rust_log,
|
has_rust_log,
|
||||||
@@ -819,3 +798,39 @@ async fn run_inner(
|
|||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(unix)]
|
||||||
|
async fn run_inner(
|
||||||
|
daemon_opts: DaemonOptions,
|
||||||
|
) -> std::result::Result<(), Box<dyn std::error::Error>> {
|
||||||
|
// Acquire PID file if daemonizing or if explicitly requested
|
||||||
|
// Keep it alive until shutdown (underscore prefix = intentionally kept for RAII cleanup)
|
||||||
|
let _pid_file = if daemon_opts.daemonize || daemon_opts.pid_file.is_some() {
|
||||||
|
let mut pf = PidFile::new(daemon_opts.pid_file_path());
|
||||||
|
if let Err(e) = pf.acquire() {
|
||||||
|
eprintln!("[telemt] {}", e);
|
||||||
|
std::process::exit(1);
|
||||||
|
}
|
||||||
|
Some(pf)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
|
let user = daemon_opts.user.clone();
|
||||||
|
let group = daemon_opts.group.clone();
|
||||||
|
|
||||||
|
run_telemt_core(|| {
|
||||||
|
if user.is_some() || group.is_some() {
|
||||||
|
if let Err(e) = drop_privileges(user.as_deref(), group.as_deref(), _pid_file.as_ref()) {
|
||||||
|
error!(error = %e, "Failed to drop privileges");
|
||||||
|
std::process::exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(not(unix))]
|
||||||
|
async fn run_inner() -> std::result::Result<(), Box<dyn std::error::Error>> {
|
||||||
|
run_telemt_core(|| {}).await
|
||||||
|
}
|
||||||
|
|||||||
@@ -97,6 +97,7 @@ pub async fn run_probe(
|
|||||||
let UpstreamType::Direct {
|
let UpstreamType::Direct {
|
||||||
interface,
|
interface,
|
||||||
bind_addresses,
|
bind_addresses,
|
||||||
|
..
|
||||||
} = &upstream.upstream_type
|
} = &upstream.upstream_type
|
||||||
else {
|
else {
|
||||||
continue;
|
continue;
|
||||||
|
|||||||
@@ -31,11 +31,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -27,11 +27,14 @@ fn build_harness(config: ProxyConfig) -> PipelineHarness {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -11,11 +11,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -11,11 +11,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -25,11 +25,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -11,11 +11,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -11,11 +11,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -38,11 +38,14 @@ fn build_harness(secret_hex: &str, mask_port: u16) -> PipelineHarness {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -16,11 +16,14 @@ fn make_test_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -39,11 +39,14 @@ fn build_harness(secret_hex: &str, mask_port: u16) -> RedTeamHarness {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -229,11 +232,14 @@ async fn redteam_03_masking_duration_must_be_less_than_1ms_when_backend_down() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -470,11 +476,14 @@ async fn measure_invalid_probe_duration_ms(delay_ms: u64, tls_len: u16, body_sen
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -544,11 +553,14 @@ async fn capture_forwarded_probe_len(tls_len: u16, body_sent: usize) -> usize {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -13,11 +13,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -11,11 +11,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -11,11 +11,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -11,11 +11,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -11,11 +11,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -25,11 +25,14 @@ fn new_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -332,11 +332,14 @@ async fn relay_task_abort_releases_user_gate_and_ip_reservation() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -446,11 +449,14 @@ async fn relay_cutover_releases_user_gate_and_ip_reservation() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -570,11 +576,14 @@ async fn integration_route_cutover_and_quota_overlap_fails_closed_and_releases_s
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -740,11 +749,14 @@ async fn proxy_protocol_header_is_rejected_when_trust_list_is_empty() {
|
|||||||
upstream_type: crate::config::UpstreamType::Direct {
|
upstream_type: crate::config::UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -817,11 +829,14 @@ async fn proxy_protocol_header_from_untrusted_peer_range_is_rejected_under_load(
|
|||||||
upstream_type: crate::config::UpstreamType::Direct {
|
upstream_type: crate::config::UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -977,11 +992,14 @@ async fn short_tls_probe_is_masked_through_client_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1065,11 +1083,14 @@ async fn tls12_record_probe_is_masked_through_client_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1151,11 +1172,14 @@ async fn handle_client_stream_increments_connects_all_exactly_once() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1244,11 +1268,14 @@ async fn running_client_handler_increments_connects_all_exactly_once() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1334,11 +1361,14 @@ async fn idle_pooled_connection_closes_cleanly_in_generic_stream_path() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1405,11 +1435,14 @@ async fn idle_pooled_connection_closes_cleanly_in_client_handler_path() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1491,11 +1524,14 @@ async fn partial_tls_header_stall_triggers_handshake_timeout() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1816,11 +1852,14 @@ async fn valid_tls_path_does_not_fall_back_to_mask_backend() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1925,11 +1964,14 @@ async fn valid_tls_with_invalid_mtproto_falls_back_to_mask_backend() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -2032,11 +2074,14 @@ async fn client_handler_tls_bad_mtproto_is_forwarded_to_mask_backend() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -2154,11 +2199,14 @@ async fn alpn_mismatch_tls_probe_is_masked_through_client_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -2247,11 +2295,14 @@ async fn invalid_hmac_tls_probe_is_masked_through_client_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -2346,11 +2397,14 @@ async fn burst_invalid_tls_probes_are_masked_verbatim() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -3251,11 +3305,14 @@ async fn relay_connect_error_releases_user_and_ip_before_return() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -3812,11 +3869,14 @@ async fn untrusted_proxy_header_source_is_rejected() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -3882,11 +3942,14 @@ async fn empty_proxy_trusted_cidrs_rejects_proxy_header_by_default() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -3979,11 +4042,14 @@ async fn oversized_tls_record_is_masked_in_generic_stream_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -4082,11 +4148,14 @@ async fn oversized_tls_record_is_masked_in_client_handler_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -4199,11 +4268,14 @@ async fn tls_record_len_min_minus_1_is_rejected_in_generic_stream_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -4302,11 +4374,14 @@ async fn tls_record_len_min_minus_1_is_rejected_in_client_handler_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -4408,11 +4483,14 @@ async fn tls_record_len_16384_is_accepted_in_generic_stream_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -4509,11 +4587,14 @@ async fn tls_record_len_16384_is_accepted_in_client_handler_pipeline() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -24,11 +24,14 @@ fn make_test_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -26,11 +26,14 @@ fn make_test_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -27,11 +27,14 @@ fn make_test_upstream_manager(stats: Arc<Stats>) -> Arc<UpstreamManager> {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -41,11 +41,14 @@ fn build_harness(secret_hex: &str, mask_port: u16) -> PipelineHarness {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -1293,11 +1293,14 @@ async fn direct_relay_abort_midflight_releases_route_gauge() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1400,11 +1403,14 @@ async fn direct_relay_cutover_midflight_releases_route_gauge() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1522,11 +1528,14 @@ async fn direct_relay_cutover_storm_multi_session_keeps_generic_errors_and_relea
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
@@ -1758,8 +1767,11 @@ async fn negative_direct_relay_dc_connection_refused_fails_fast() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
100,
|
100,
|
||||||
@@ -1849,8 +1861,11 @@ async fn adversarial_direct_relay_cutover_integrity() {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
100,
|
100,
|
||||||
|
|||||||
@@ -53,11 +53,14 @@ fn new_client_harness() -> ClientHarness {
|
|||||||
upstream_type: UpstreamType::Direct {
|
upstream_type: UpstreamType::Direct {
|
||||||
interface: None,
|
interface: None,
|
||||||
bind_addresses: None,
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
},
|
},
|
||||||
weight: 1,
|
weight: 1,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
1,
|
1,
|
||||||
|
|||||||
@@ -67,6 +67,7 @@ pub fn format_sample_line(sample: &MePingSample) -> String {
|
|||||||
fn format_direct_with_config(
|
fn format_direct_with_config(
|
||||||
interface: &Option<String>,
|
interface: &Option<String>,
|
||||||
bind_addresses: &Option<Vec<String>>,
|
bind_addresses: &Option<Vec<String>>,
|
||||||
|
bindtodevice: &Option<String>,
|
||||||
) -> Option<String> {
|
) -> Option<String> {
|
||||||
let mut direct_parts: Vec<String> = Vec::new();
|
let mut direct_parts: Vec<String> = Vec::new();
|
||||||
if let Some(dev) = interface.as_deref().filter(|v| !v.is_empty()) {
|
if let Some(dev) = interface.as_deref().filter(|v| !v.is_empty()) {
|
||||||
@@ -75,6 +76,9 @@ fn format_direct_with_config(
|
|||||||
if let Some(src) = bind_addresses.as_ref().filter(|v| !v.is_empty()) {
|
if let Some(src) = bind_addresses.as_ref().filter(|v| !v.is_empty()) {
|
||||||
direct_parts.push(format!("src={}", src.join(",")));
|
direct_parts.push(format!("src={}", src.join(",")));
|
||||||
}
|
}
|
||||||
|
if let Some(device) = bindtodevice.as_deref().filter(|v| !v.is_empty()) {
|
||||||
|
direct_parts.push(format!("bindtodevice={device}"));
|
||||||
|
}
|
||||||
if direct_parts.is_empty() {
|
if direct_parts.is_empty() {
|
||||||
None
|
None
|
||||||
} else {
|
} else {
|
||||||
@@ -231,8 +235,11 @@ pub async fn format_me_route(
|
|||||||
UpstreamType::Direct {
|
UpstreamType::Direct {
|
||||||
interface,
|
interface,
|
||||||
bind_addresses,
|
bind_addresses,
|
||||||
|
bindtodevice,
|
||||||
} => {
|
} => {
|
||||||
if let Some(route) = format_direct_with_config(interface, bind_addresses) {
|
if let Some(route) =
|
||||||
|
format_direct_with_config(interface, bind_addresses, bindtodevice)
|
||||||
|
{
|
||||||
route
|
route
|
||||||
} else {
|
} else {
|
||||||
detect_direct_route_details(reports, prefer_ipv6, v4_ok, v6_ok)
|
detect_direct_route_details(reports, prefer_ipv6, v4_ok, v6_ok)
|
||||||
|
|||||||
@@ -158,6 +158,56 @@ pub fn create_outgoing_socket_bound(addr: SocketAddr, bind_addr: Option<IpAddr>)
|
|||||||
Ok(socket)
|
Ok(socket)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Pin an outgoing socket to a specific Linux network interface via SO_BINDTODEVICE.
|
||||||
|
#[cfg(target_os = "linux")]
|
||||||
|
pub fn bind_outgoing_socket_to_device(socket: &Socket, device: &str) -> Result<()> {
|
||||||
|
use std::io::{Error, ErrorKind};
|
||||||
|
use std::os::fd::AsRawFd;
|
||||||
|
|
||||||
|
let name = device.trim();
|
||||||
|
if name.is_empty() {
|
||||||
|
return Err(Error::new(
|
||||||
|
ErrorKind::InvalidInput,
|
||||||
|
"bindtodevice must not be empty",
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
// The kernel expects an interface name buffer with a trailing NUL.
|
||||||
|
if name.len() >= libc::IFNAMSIZ {
|
||||||
|
return Err(Error::new(
|
||||||
|
ErrorKind::InvalidInput,
|
||||||
|
"bindtodevice exceeds IFNAMSIZ",
|
||||||
|
));
|
||||||
|
}
|
||||||
|
let mut ifname = [0u8; libc::IFNAMSIZ];
|
||||||
|
ifname[..name.len()].copy_from_slice(name.as_bytes());
|
||||||
|
|
||||||
|
let rc = unsafe {
|
||||||
|
libc::setsockopt(
|
||||||
|
socket.as_raw_fd(),
|
||||||
|
libc::SOL_SOCKET,
|
||||||
|
libc::SO_BINDTODEVICE,
|
||||||
|
ifname.as_ptr().cast::<libc::c_void>(),
|
||||||
|
(name.len() + 1) as libc::socklen_t,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
if rc != 0 {
|
||||||
|
return Err(Error::last_os_error());
|
||||||
|
}
|
||||||
|
debug!("Pinned outgoing socket to interface {}", name);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Stub for non-Linux targets where SO_BINDTODEVICE is unavailable.
|
||||||
|
#[cfg(not(target_os = "linux"))]
|
||||||
|
pub fn bind_outgoing_socket_to_device(_socket: &Socket, _device: &str) -> Result<()> {
|
||||||
|
use std::io::{Error, ErrorKind};
|
||||||
|
Err(Error::new(
|
||||||
|
ErrorKind::Unsupported,
|
||||||
|
"bindtodevice is supported only on Linux",
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
/// Get local address of a socket
|
/// Get local address of a socket
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub fn get_local_addr(stream: &TcpStream) -> Option<SocketAddr> {
|
pub fn get_local_addr(stream: &TcpStream) -> Option<SocketAddr> {
|
||||||
|
|||||||
+234
-26
@@ -26,7 +26,9 @@ use crate::stats::Stats;
|
|||||||
use crate::transport::shadowsocks::{
|
use crate::transport::shadowsocks::{
|
||||||
ShadowsocksStream, connect_shadowsocks, sanitize_shadowsocks_url,
|
ShadowsocksStream, connect_shadowsocks, sanitize_shadowsocks_url,
|
||||||
};
|
};
|
||||||
use crate::transport::socket::{create_outgoing_socket_bound, resolve_interface_ip};
|
use crate::transport::socket::{
|
||||||
|
bind_outgoing_socket_to_device, create_outgoing_socket_bound, resolve_interface_ip,
|
||||||
|
};
|
||||||
use crate::transport::socks::{connect_socks4, connect_socks5};
|
use crate::transport::socks::{connect_socks4, connect_socks5};
|
||||||
|
|
||||||
/// Number of Telegram datacenters
|
/// Number of Telegram datacenters
|
||||||
@@ -327,6 +329,17 @@ pub struct UpstreamManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl UpstreamManager {
|
impl UpstreamManager {
|
||||||
|
fn is_unscoped_upstream(upstream: &UpstreamConfig) -> bool {
|
||||||
|
upstream.scopes.is_empty()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn should_check_in_default_dc_connectivity(
|
||||||
|
has_unscoped: bool,
|
||||||
|
upstream: &UpstreamConfig,
|
||||||
|
) -> bool {
|
||||||
|
!has_unscoped || Self::is_unscoped_upstream(upstream)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn new(
|
pub fn new(
|
||||||
configs: Vec<UpstreamConfig>,
|
configs: Vec<UpstreamConfig>,
|
||||||
connect_retry_attempts: u32,
|
connect_retry_attempts: u32,
|
||||||
@@ -453,6 +466,87 @@ impl UpstreamManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn resolve_probe_dc_families(
|
||||||
|
upstream: &UpstreamConfig,
|
||||||
|
ipv4_available: bool,
|
||||||
|
ipv6_available: bool,
|
||||||
|
) -> (bool, bool) {
|
||||||
|
(
|
||||||
|
upstream.ipv4.unwrap_or(ipv4_available),
|
||||||
|
upstream.ipv6.unwrap_or(ipv6_available),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn resolve_runtime_dc_families(
|
||||||
|
upstream: &UpstreamConfig,
|
||||||
|
dc_preference: IpPreference,
|
||||||
|
) -> (bool, bool) {
|
||||||
|
let (auto_ipv4, auto_ipv6) = match dc_preference {
|
||||||
|
IpPreference::PreferV4 => (true, false),
|
||||||
|
IpPreference::PreferV6 => (false, true),
|
||||||
|
IpPreference::BothWork | IpPreference::Unknown | IpPreference::Unavailable => {
|
||||||
|
(true, true)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
(
|
||||||
|
upstream.ipv4.unwrap_or(auto_ipv4),
|
||||||
|
upstream.ipv6.unwrap_or(auto_ipv6),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn dc_table_addr(dc_idx: i16, ipv6: bool, port: u16) -> Option<SocketAddr> {
|
||||||
|
let arr_idx = UpstreamState::dc_array_idx(dc_idx)?;
|
||||||
|
let ip = if ipv6 {
|
||||||
|
TG_DATACENTERS_V6[arr_idx]
|
||||||
|
} else {
|
||||||
|
TG_DATACENTERS_V4[arr_idx]
|
||||||
|
};
|
||||||
|
Some(SocketAddr::new(ip, port))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn resolve_runtime_dc_target(
|
||||||
|
target: SocketAddr,
|
||||||
|
dc_idx: Option<i16>,
|
||||||
|
upstream: &UpstreamConfig,
|
||||||
|
dc_preference: IpPreference,
|
||||||
|
) -> Result<SocketAddr> {
|
||||||
|
let (allow_ipv4, allow_ipv6) = Self::resolve_runtime_dc_families(upstream, dc_preference);
|
||||||
|
if (target.is_ipv4() && allow_ipv4) || (target.is_ipv6() && allow_ipv6) {
|
||||||
|
return Ok(target);
|
||||||
|
}
|
||||||
|
|
||||||
|
if !allow_ipv4 && !allow_ipv6 {
|
||||||
|
return Err(ProxyError::Config(format!(
|
||||||
|
"Upstream DC family policy blocks all families for target {target}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
|
||||||
|
let Some(dc_idx) = dc_idx else {
|
||||||
|
return Err(ProxyError::Config(format!(
|
||||||
|
"Upstream DC family policy cannot remap target {target} without dc_idx"
|
||||||
|
)));
|
||||||
|
};
|
||||||
|
|
||||||
|
let remapped = if target.is_ipv4() {
|
||||||
|
if allow_ipv6 {
|
||||||
|
Self::dc_table_addr(dc_idx, true, target.port())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
} else if allow_ipv4 {
|
||||||
|
Self::dc_table_addr(dc_idx, false, target.port())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
|
remapped.ok_or_else(|| {
|
||||||
|
ProxyError::Config(format!(
|
||||||
|
"Upstream DC family policy rejected target {target} (dc_idx={dc_idx})"
|
||||||
|
))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(unix)]
|
#[cfg(unix)]
|
||||||
fn resolve_interface_addrs(name: &str, want_ipv6: bool) -> Vec<IpAddr> {
|
fn resolve_interface_addrs(name: &str, want_ipv6: bool) -> Vec<IpAddr> {
|
||||||
use nix::ifaddrs::getifaddrs;
|
use nix::ifaddrs::getifaddrs;
|
||||||
@@ -726,18 +820,28 @@ impl UpstreamManager {
|
|||||||
.await
|
.await
|
||||||
.ok_or_else(|| ProxyError::Config("No upstreams available".to_string()))?;
|
.ok_or_else(|| ProxyError::Config("No upstreams available".to_string()))?;
|
||||||
|
|
||||||
let mut upstream = {
|
let (mut upstream, bind_rr, dc_preference) = {
|
||||||
let guard = self.upstreams.read().await;
|
let guard = self.upstreams.read().await;
|
||||||
guard[idx].config.clone()
|
let state = &guard[idx];
|
||||||
|
let dc_preference = dc_idx
|
||||||
|
.and_then(UpstreamState::dc_array_idx)
|
||||||
|
.map(|dc_array_idx| state.dc_ip_pref[dc_array_idx])
|
||||||
|
.unwrap_or(IpPreference::Unknown);
|
||||||
|
(
|
||||||
|
state.config.clone(),
|
||||||
|
Some(state.bind_rr.clone()),
|
||||||
|
dc_preference,
|
||||||
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
if let Some(s) = scope {
|
if let Some(s) = scope {
|
||||||
upstream.selected_scope = s.to_string();
|
upstream.selected_scope = s.to_string();
|
||||||
}
|
}
|
||||||
|
|
||||||
let bind_rr = {
|
let target = if dc_idx.is_some() {
|
||||||
let guard = self.upstreams.read().await;
|
Self::resolve_runtime_dc_target(target, dc_idx, &upstream, dc_preference)?
|
||||||
guard.get(idx).map(|u| u.bind_rr.clone())
|
} else {
|
||||||
|
target
|
||||||
};
|
};
|
||||||
|
|
||||||
let (stream, _) = self
|
let (stream, _) = self
|
||||||
@@ -758,9 +862,18 @@ impl UpstreamManager {
|
|||||||
.await
|
.await
|
||||||
.ok_or_else(|| ProxyError::Config("No upstreams available".to_string()))?;
|
.ok_or_else(|| ProxyError::Config("No upstreams available".to_string()))?;
|
||||||
|
|
||||||
let mut upstream = {
|
let (mut upstream, bind_rr, dc_preference) = {
|
||||||
let guard = self.upstreams.read().await;
|
let guard = self.upstreams.read().await;
|
||||||
guard[idx].config.clone()
|
let state = &guard[idx];
|
||||||
|
let dc_preference = dc_idx
|
||||||
|
.and_then(UpstreamState::dc_array_idx)
|
||||||
|
.map(|dc_array_idx| state.dc_ip_pref[dc_array_idx])
|
||||||
|
.unwrap_or(IpPreference::Unknown);
|
||||||
|
(
|
||||||
|
state.config.clone(),
|
||||||
|
Some(state.bind_rr.clone()),
|
||||||
|
dc_preference,
|
||||||
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
// Set scope for configuration copy
|
// Set scope for configuration copy
|
||||||
@@ -768,9 +881,10 @@ impl UpstreamManager {
|
|||||||
upstream.selected_scope = s.to_string();
|
upstream.selected_scope = s.to_string();
|
||||||
}
|
}
|
||||||
|
|
||||||
let bind_rr = {
|
let target = if dc_idx.is_some() {
|
||||||
let guard = self.upstreams.read().await;
|
Self::resolve_runtime_dc_target(target, dc_idx, &upstream, dc_preference)?
|
||||||
guard.get(idx).map(|u| u.bind_rr.clone())
|
} else {
|
||||||
|
target
|
||||||
};
|
};
|
||||||
|
|
||||||
let (stream, egress) = self
|
let (stream, egress) = self
|
||||||
@@ -928,6 +1042,7 @@ impl UpstreamManager {
|
|||||||
UpstreamType::Direct {
|
UpstreamType::Direct {
|
||||||
interface,
|
interface,
|
||||||
bind_addresses,
|
bind_addresses,
|
||||||
|
bindtodevice,
|
||||||
} => {
|
} => {
|
||||||
let bind_ip = Self::resolve_bind_address(
|
let bind_ip = Self::resolve_bind_address(
|
||||||
interface,
|
interface,
|
||||||
@@ -943,6 +1058,10 @@ impl UpstreamManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let socket = create_outgoing_socket_bound(target, bind_ip)?;
|
let socket = create_outgoing_socket_bound(target, bind_ip)?;
|
||||||
|
if let Some(device) = bindtodevice.as_deref().filter(|value| !value.is_empty()) {
|
||||||
|
bind_outgoing_socket_to_device(&socket, device).map_err(ProxyError::Io)?;
|
||||||
|
debug!(bindtodevice = %device, target = %target, "Pinned socket to interface");
|
||||||
|
}
|
||||||
if let Some(ip) = bind_ip {
|
if let Some(ip) = bind_ip {
|
||||||
debug!(bind = %ip, target = %target, "Bound outgoing socket");
|
debug!(bind = %ip, target = %target, "Bound outgoing socket");
|
||||||
} else if interface.is_some() || bind_addresses.is_some() {
|
} else if interface.is_some() || bind_addresses.is_some() {
|
||||||
@@ -1201,14 +1320,26 @@ impl UpstreamManager {
|
|||||||
.map(|(i, u)| (i, u.config.clone(), u.bind_rr.clone()))
|
.map(|(i, u)| (i, u.config.clone(), u.bind_rr.clone()))
|
||||||
.collect()
|
.collect()
|
||||||
};
|
};
|
||||||
|
let has_unscoped = upstreams
|
||||||
|
.iter()
|
||||||
|
.any(|(_, cfg, _)| Self::is_unscoped_upstream(cfg));
|
||||||
|
|
||||||
let mut all_results = Vec::new();
|
let mut all_results = Vec::new();
|
||||||
|
|
||||||
for (upstream_idx, upstream_config, bind_rr) in &upstreams {
|
for (upstream_idx, upstream_config, bind_rr) in &upstreams {
|
||||||
|
// DC connectivity checks should follow the default routing path.
|
||||||
|
// Scoped upstreams are included only when no unscoped upstream exists.
|
||||||
|
if !Self::should_check_in_default_dc_connectivity(has_unscoped, upstream_config) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let (upstream_ipv4_enabled, upstream_ipv6_enabled) =
|
||||||
|
Self::resolve_probe_dc_families(upstream_config, ipv4_enabled, ipv6_enabled);
|
||||||
let upstream_name = match &upstream_config.upstream_type {
|
let upstream_name = match &upstream_config.upstream_type {
|
||||||
UpstreamType::Direct {
|
UpstreamType::Direct {
|
||||||
interface,
|
interface,
|
||||||
bind_addresses,
|
bind_addresses,
|
||||||
|
bindtodevice,
|
||||||
} => {
|
} => {
|
||||||
let mut direct_parts = Vec::new();
|
let mut direct_parts = Vec::new();
|
||||||
if let Some(dev) = interface.as_deref().filter(|v| !v.is_empty()) {
|
if let Some(dev) = interface.as_deref().filter(|v| !v.is_empty()) {
|
||||||
@@ -1217,6 +1348,9 @@ impl UpstreamManager {
|
|||||||
if let Some(src) = bind_addresses.as_ref().filter(|v| !v.is_empty()) {
|
if let Some(src) = bind_addresses.as_ref().filter(|v| !v.is_empty()) {
|
||||||
direct_parts.push(format!("src={}", src.join(",")));
|
direct_parts.push(format!("src={}", src.join(",")));
|
||||||
}
|
}
|
||||||
|
if let Some(device) = bindtodevice.as_deref().filter(|v| !v.is_empty()) {
|
||||||
|
direct_parts.push(format!("bindtodevice={device}"));
|
||||||
|
}
|
||||||
if direct_parts.is_empty() {
|
if direct_parts.is_empty() {
|
||||||
"direct".to_string()
|
"direct".to_string()
|
||||||
} else {
|
} else {
|
||||||
@@ -1233,7 +1367,7 @@ impl UpstreamManager {
|
|||||||
};
|
};
|
||||||
|
|
||||||
let mut v6_results = Vec::with_capacity(NUM_DCS);
|
let mut v6_results = Vec::with_capacity(NUM_DCS);
|
||||||
if ipv6_enabled {
|
if upstream_ipv6_enabled {
|
||||||
for dc_zero_idx in 0..NUM_DCS {
|
for dc_zero_idx in 0..NUM_DCS {
|
||||||
let dc_v6 = TG_DATACENTERS_V6[dc_zero_idx];
|
let dc_v6 = TG_DATACENTERS_V6[dc_zero_idx];
|
||||||
let addr_v6 = SocketAddr::new(dc_v6, TG_DATACENTER_PORT);
|
let addr_v6 = SocketAddr::new(dc_v6, TG_DATACENTER_PORT);
|
||||||
@@ -1284,13 +1418,17 @@ impl UpstreamManager {
|
|||||||
dc_idx: dc_zero_idx + 1,
|
dc_idx: dc_zero_idx + 1,
|
||||||
dc_addr: SocketAddr::new(dc_v6, TG_DATACENTER_PORT),
|
dc_addr: SocketAddr::new(dc_v6, TG_DATACENTER_PORT),
|
||||||
rtt_ms: None,
|
rtt_ms: None,
|
||||||
error: Some("ipv6 disabled".to_string()),
|
error: Some(if ipv6_enabled {
|
||||||
|
"ipv6 disabled by upstream policy".to_string()
|
||||||
|
} else {
|
||||||
|
"ipv6 disabled".to_string()
|
||||||
|
}),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut v4_results = Vec::with_capacity(NUM_DCS);
|
let mut v4_results = Vec::with_capacity(NUM_DCS);
|
||||||
if ipv4_enabled {
|
if upstream_ipv4_enabled {
|
||||||
for dc_zero_idx in 0..NUM_DCS {
|
for dc_zero_idx in 0..NUM_DCS {
|
||||||
let dc_v4 = TG_DATACENTERS_V4[dc_zero_idx];
|
let dc_v4 = TG_DATACENTERS_V4[dc_zero_idx];
|
||||||
let addr_v4 = SocketAddr::new(dc_v4, TG_DATACENTER_PORT);
|
let addr_v4 = SocketAddr::new(dc_v4, TG_DATACENTER_PORT);
|
||||||
@@ -1341,7 +1479,11 @@ impl UpstreamManager {
|
|||||||
dc_idx: dc_zero_idx + 1,
|
dc_idx: dc_zero_idx + 1,
|
||||||
dc_addr: SocketAddr::new(dc_v4, TG_DATACENTER_PORT),
|
dc_addr: SocketAddr::new(dc_v4, TG_DATACENTER_PORT),
|
||||||
rtt_ms: None,
|
rtt_ms: None,
|
||||||
error: Some("ipv4 disabled".to_string()),
|
error: Some(if ipv4_enabled {
|
||||||
|
"ipv4 disabled by upstream policy".to_string()
|
||||||
|
} else {
|
||||||
|
"ipv4 disabled".to_string()
|
||||||
|
}),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1361,7 +1503,9 @@ impl UpstreamManager {
|
|||||||
match addr_str.parse::<SocketAddr>() {
|
match addr_str.parse::<SocketAddr>() {
|
||||||
Ok(addr) => {
|
Ok(addr) => {
|
||||||
let is_v6 = addr.is_ipv6();
|
let is_v6 = addr.is_ipv6();
|
||||||
if (is_v6 && !ipv6_enabled) || (!is_v6 && !ipv4_enabled) {
|
if (is_v6 && !upstream_ipv6_enabled)
|
||||||
|
|| (!is_v6 && !upstream_ipv4_enabled)
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
let result = tokio::time::timeout(
|
let result = tokio::time::timeout(
|
||||||
@@ -1596,13 +1740,32 @@ impl UpstreamManager {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
let count = self.upstreams.read().await.len();
|
let target_upstreams: Vec<usize> = {
|
||||||
for i in 0..count {
|
let guard = self.upstreams.read().await;
|
||||||
|
let has_unscoped = guard
|
||||||
|
.iter()
|
||||||
|
.any(|upstream| Self::is_unscoped_upstream(&upstream.config));
|
||||||
|
guard
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.filter(|(_, upstream)| {
|
||||||
|
Self::should_check_in_default_dc_connectivity(
|
||||||
|
has_unscoped,
|
||||||
|
&upstream.config,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.map(|(idx, _)| idx)
|
||||||
|
.collect()
|
||||||
|
};
|
||||||
|
|
||||||
|
for i in target_upstreams {
|
||||||
let (config, bind_rr) = {
|
let (config, bind_rr) = {
|
||||||
let guard = self.upstreams.read().await;
|
let guard = self.upstreams.read().await;
|
||||||
let u = &guard[i];
|
let u = &guard[i];
|
||||||
(u.config.clone(), u.bind_rr.clone())
|
(u.config.clone(), u.bind_rr.clone())
|
||||||
};
|
};
|
||||||
|
let (upstream_ipv4_enabled, upstream_ipv6_enabled) =
|
||||||
|
Self::resolve_probe_dc_families(&config, ipv4_enabled, ipv6_enabled);
|
||||||
|
|
||||||
let mut healthy_groups = 0usize;
|
let mut healthy_groups = 0usize;
|
||||||
let mut latency_updates: Vec<(usize, f64)> = Vec::new();
|
let mut latency_updates: Vec<(usize, f64)> = Vec::new();
|
||||||
@@ -1618,14 +1781,30 @@ impl UpstreamManager {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
let rotation_key = (i, group.dc_idx, is_primary);
|
let filtered_endpoints: Vec<SocketAddr> = endpoints
|
||||||
let start_idx =
|
.iter()
|
||||||
*endpoint_rotation.entry(rotation_key).or_insert(0) % endpoints.len();
|
.copied()
|
||||||
let mut next_idx = (start_idx + 1) % endpoints.len();
|
.filter(|endpoint| {
|
||||||
|
if endpoint.is_ipv4() {
|
||||||
|
upstream_ipv4_enabled
|
||||||
|
} else {
|
||||||
|
upstream_ipv6_enabled
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
for step in 0..endpoints.len() {
|
if filtered_endpoints.is_empty() {
|
||||||
let endpoint_idx = (start_idx + step) % endpoints.len();
|
continue;
|
||||||
let endpoint = endpoints[endpoint_idx];
|
}
|
||||||
|
|
||||||
|
let rotation_key = (i, group.dc_idx, is_primary);
|
||||||
|
let start_idx = *endpoint_rotation.entry(rotation_key).or_insert(0)
|
||||||
|
% filtered_endpoints.len();
|
||||||
|
let mut next_idx = (start_idx + 1) % filtered_endpoints.len();
|
||||||
|
|
||||||
|
for step in 0..filtered_endpoints.len() {
|
||||||
|
let endpoint_idx = (start_idx + step) % filtered_endpoints.len();
|
||||||
|
let endpoint = filtered_endpoints[endpoint_idx];
|
||||||
|
|
||||||
let start = Instant::now();
|
let start = Instant::now();
|
||||||
let result = tokio::time::timeout(
|
let result = tokio::time::timeout(
|
||||||
@@ -1644,7 +1823,7 @@ impl UpstreamManager {
|
|||||||
Ok(Ok(_stream)) => {
|
Ok(Ok(_stream)) => {
|
||||||
group_ok = true;
|
group_ok = true;
|
||||||
group_rtt_ms = Some(start.elapsed().as_secs_f64() * 1000.0);
|
group_rtt_ms = Some(start.elapsed().as_secs_f64() * 1000.0);
|
||||||
next_idx = (endpoint_idx + 1) % endpoints.len();
|
next_idx = (endpoint_idx + 1) % filtered_endpoints.len();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
Ok(Err(e)) => {
|
Ok(Err(e)) => {
|
||||||
@@ -1859,6 +2038,33 @@ mod tests {
|
|||||||
assert!(!UpstreamManager::is_hard_connect_error(&error));
|
assert!(!UpstreamManager::is_hard_connect_error(&error));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn unscoped_selection_detects_default_route_upstream() {
|
||||||
|
let mut upstream = UpstreamConfig {
|
||||||
|
upstream_type: UpstreamType::Direct {
|
||||||
|
interface: None,
|
||||||
|
bind_addresses: None,
|
||||||
|
bindtodevice: None,
|
||||||
|
},
|
||||||
|
weight: 1,
|
||||||
|
enabled: true,
|
||||||
|
scopes: String::new(),
|
||||||
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
assert!(UpstreamManager::is_unscoped_upstream(&upstream));
|
||||||
|
upstream.scopes = "local".to_string();
|
||||||
|
assert!(!UpstreamManager::is_unscoped_upstream(&upstream));
|
||||||
|
assert!(!UpstreamManager::should_check_in_default_dc_connectivity(
|
||||||
|
true, &upstream
|
||||||
|
));
|
||||||
|
assert!(UpstreamManager::should_check_in_default_dc_connectivity(
|
||||||
|
false, &upstream
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn resolve_bind_address_prefers_explicit_bind_ip() {
|
fn resolve_bind_address_prefers_explicit_bind_ip() {
|
||||||
let target = "203.0.113.10:443".parse::<SocketAddr>().unwrap();
|
let target = "203.0.113.10:443".parse::<SocketAddr>().unwrap();
|
||||||
@@ -1899,6 +2105,8 @@ mod tests {
|
|||||||
enabled: true,
|
enabled: true,
|
||||||
scopes: String::new(),
|
scopes: String::new(),
|
||||||
selected_scope: String::new(),
|
selected_scope: String::new(),
|
||||||
|
ipv4: None,
|
||||||
|
ipv6: None,
|
||||||
}],
|
}],
|
||||||
1,
|
1,
|
||||||
100,
|
100,
|
||||||
|
|||||||
Reference in New Issue
Block a user