mirror of
https://github.com/by-sonic/tglock.git
synced 2026-05-22 15:31:42 +03:00
Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6795c6177a | ||
|
|
70c61e1330 | ||
|
|
09b7a03a0a |
77
.github/workflows/release.yml
vendored
77
.github/workflows/release.yml
vendored
@@ -1,64 +1,83 @@
|
||||
name: Build & Release
|
||||
name: Release
|
||||
|
||||
on:
|
||||
push:
|
||||
tags:
|
||||
- 'v*'
|
||||
tags: ['v*']
|
||||
|
||||
permissions:
|
||||
contents: write
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: windows-latest
|
||||
strategy:
|
||||
matrix:
|
||||
include:
|
||||
- os: windows-latest
|
||||
target: x86_64-pc-windows-msvc
|
||||
artifact: tglock.exe
|
||||
- os: macos-latest
|
||||
target: x86_64-apple-darwin
|
||||
artifact: tglock-macos-x64
|
||||
- os: macos-latest
|
||||
target: aarch64-apple-darwin
|
||||
artifact: tglock-macos-arm64
|
||||
- os: ubuntu-latest
|
||||
target: x86_64-unknown-linux-gnu
|
||||
artifact: tglock-linux-x64
|
||||
|
||||
runs-on: ${{ matrix.os }}
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Install Rust
|
||||
uses: dtolnay/rust-toolchain@stable
|
||||
|
||||
- name: Cache cargo
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: |
|
||||
~/.cargo/registry
|
||||
~/.cargo/git
|
||||
target
|
||||
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
|
||||
targets: ${{ matrix.target }}
|
||||
|
||||
- name: Build release
|
||||
run: cargo build --release
|
||||
- name: Build
|
||||
run: cargo build --release --target ${{ matrix.target }}
|
||||
|
||||
- name: Upload artifact
|
||||
- name: Rename (Unix)
|
||||
if: runner.os != 'Windows'
|
||||
run: cp target/${{ matrix.target }}/release/tglock ${{ matrix.artifact }}
|
||||
|
||||
- name: Rename (Windows)
|
||||
if: runner.os == 'Windows'
|
||||
run: copy target\${{ matrix.target }}\release\tglock.exe ${{ matrix.artifact }}
|
||||
|
||||
- name: Upload
|
||||
uses: actions/upload-artifact@v4
|
||||
with:
|
||||
name: tg_unblock-windows-x64
|
||||
path: target/release/tg_unblock.exe
|
||||
name: ${{ matrix.artifact }}
|
||||
path: ${{ matrix.artifact }}
|
||||
|
||||
release:
|
||||
needs: build
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Download artifact
|
||||
- name: Download artifacts
|
||||
uses: actions/download-artifact@v4
|
||||
with:
|
||||
name: tg_unblock-windows-x64
|
||||
|
||||
- name: Create Release
|
||||
uses: softprops/action-gh-release@v2
|
||||
with:
|
||||
files: tg_unblock.exe
|
||||
generate_release_notes: true
|
||||
files: |
|
||||
tglock.exe/tglock.exe
|
||||
tglock-macos-x64/tglock-macos-x64
|
||||
tglock-macos-arm64/tglock-macos-arm64
|
||||
tglock-linux-x64/tglock-linux-x64
|
||||
body: |
|
||||
## TG Unblock ${{ github.ref_name }}
|
||||
## TGLock ${{ github.ref_name }}
|
||||
|
||||
Обход блокировки Telegram через WebSocket-туннель.
|
||||
Обход блокировки Telegram через WebSocket-туннель. Один клик.
|
||||
|
||||
### Установка
|
||||
1. Скачайте `tg_unblock.exe`
|
||||
2. Запустите
|
||||
3. Нажмите "Запустить обход"
|
||||
4. Нажмите "Настроить автоматически"
|
||||
| Файл | Платформа |
|
||||
|---|---|
|
||||
| `tglock.exe` | Windows x64 |
|
||||
| `tglock-macos-x64` | macOS Intel |
|
||||
| `tglock-macos-arm64` | macOS Apple Silicon |
|
||||
| `tglock-linux-x64` | Linux x64 |
|
||||
|
||||
---
|
||||
**[by sonic VPN](https://t.me/bysonicvpn_bot)** — полный обход блокировок для всех приложений
|
||||
|
||||
5
.gitignore
vendored
5
.gitignore
vendored
@@ -1,6 +1,5 @@
|
||||
/target
|
||||
/tools
|
||||
*.zip
|
||||
*.exe
|
||||
!*.rs
|
||||
Cargo.lock
|
||||
*.zip
|
||||
.claude/
|
||||
|
||||
18
Cargo.toml
18
Cargo.toml
@@ -1,26 +1,22 @@
|
||||
[package]
|
||||
name = "tg_unblock"
|
||||
version = "0.3.1"
|
||||
name = "tglock"
|
||||
version = "1.0.0"
|
||||
edition = "2021"
|
||||
description = "Telegram unblock via local WebSocket tunnel"
|
||||
license = "MIT"
|
||||
|
||||
[dependencies]
|
||||
eframe = "0.31"
|
||||
egui = "0.31"
|
||||
tokio = { version = "1", features = ["full"] }
|
||||
reqwest = { version = "0.12", features = ["blocking"] }
|
||||
serde = { version = "1", features = ["derive"] }
|
||||
serde_json = "1"
|
||||
open = "5"
|
||||
tokio = { version = "1", features = ["rt-multi-thread", "net", "io-util", "time", "macros"] }
|
||||
tokio-tungstenite = { version = "0.24", features = ["native-tls"] }
|
||||
native-tls = "0.2"
|
||||
futures-util = "0.3"
|
||||
aes = "0.8"
|
||||
ctr = "0.9"
|
||||
cipher = "0.4"
|
||||
|
||||
[target.'cfg(windows)'.dependencies]
|
||||
winapi = { version = "0.3", features = ["winuser"] }
|
||||
open = "5"
|
||||
|
||||
[[bin]]
|
||||
name = "tg_unblock"
|
||||
name = "tglock"
|
||||
path = "src/main.rs"
|
||||
|
||||
175
HABR.md
Normal file
175
HABR.md
Normal file
@@ -0,0 +1,175 @@
|
||||
# Написал обход блокировки Telegram на Rust за 300 строк — без VPN, серверов и абонентки
|
||||
|
||||
**Простой · 6 мин · Rust · Open source · Сетевые технологии · Из песочницы**
|
||||
|
||||
**TL;DR:** Open-source приложение **TGLock** на Rust. Один клик — Telegram работает. Локальный SOCKS5-прокси заворачивает MTProto в WebSocket через `web.telegram.org`. Провайдер видит HTTPS. Windows, macOS, Linux. 300 строк кода. GitHub — [by-sonic/tglock](https://github.com/by-sonic/tglock).
|
||||
|
||||
---
|
||||
|
||||
## Почему GoodbyeDPI больше не хватает
|
||||
|
||||
GoodbyeDPI, Zapret — отличные инструменты. Они фрагментируют пакеты, ломают сигнатуры DPI, и это работало. До определённого момента.
|
||||
|
||||
Проблема: провайдеры перешли от DPI к **IP-шейпингу**. Весь трафик к подсетям Telegram (149.154.x.x, 91.108.x.x) режется по скорости. Неважно, видит DPI MTProto или нет — если destination IP принадлежит Telegram, соединение троттлится.
|
||||
|
||||
Результат: GoodbyeDPI запущен, пакеты фрагментированы, DPI обманут — а Telegram всё равно грузится 10 секунд, медиа не приходят, звонки рвутся. Пинг 200+, постоянные переподключения.
|
||||
|
||||
VPN решает, но:
|
||||
- Стоит денег
|
||||
- Гонит **весь** трафик через чужой сервер
|
||||
- Для одного Telegram — оверкилл
|
||||
|
||||
Нужен другой подход.
|
||||
|
||||
## Идея: WebSocket через web.telegram.org
|
||||
|
||||
Замерил: прямое TCP-соединение к серверам Telegram (149.154.167.51:443) — таймаут или 200+ мс. А вот `web.telegram.org` отвечает стабильно за 50–80 мс. Логично: это «обычный сайт», провайдер его не трогает.
|
||||
|
||||
Полез в [документацию MTProto](https://core.telegram.org/mtproto/transports):
|
||||
|
||||
> **WebSocket:** Implementation of the WebSocket transport is **pretty much the same as with TCP**... all data received and sent through WebSocket messages is to be treated as a **single duplex stream of bytes**, just like with TCP.
|
||||
|
||||
Telegram официально поддерживает WebSocket-транспорт. Эндпоинты `kws1-5.web.telegram.org` — полноценные точки входа в сеть Telegram через WSS.
|
||||
|
||||
**Схема:**
|
||||
|
||||
```
|
||||
Telegram Desktop → SOCKS5 → TGLock → WSS (kws{dc}.web.telegram.org) → DC
|
||||
↑
|
||||
Провайдер видит: HTTPS к web.telegram.org
|
||||
```
|
||||
|
||||
Нет MTProto в трафике. Нет подозрительных IP. Обычный HTTPS.
|
||||
|
||||
## Реализация: 300 строк на Rust
|
||||
|
||||
Весь проект — два файла: `proxy.rs` (туннель) и `main.rs` (UI).
|
||||
|
||||
### SOCKS5 → определение DC → WebSocket
|
||||
|
||||
Когда Telegram Desktop подключается через SOCKS5, мы:
|
||||
|
||||
**1.** Обрабатываем SOCKS5-хендшейк и получаем destination IP.
|
||||
|
||||
**2.** Читаем первые 64 байта — это obfuscated2 init-пакет MTProto. Из него извлекаем настоящий DC через AES-256-CTR:
|
||||
|
||||
```rust
|
||||
fn dc_from_init(init: &[u8; 64]) -> Option<u8> {
|
||||
use aes::Aes256;
|
||||
use cipher::{KeyIvInit, StreamCipher};
|
||||
|
||||
let mut dec = *init;
|
||||
let mut c = ctr::Ctr128BE::<Aes256>::new(
|
||||
init[8..40].into(),
|
||||
init[40..56].into(),
|
||||
);
|
||||
c.apply_keystream(&mut dec);
|
||||
|
||||
let id = i32::from_le_bytes([dec[60], dec[61], dec[62], dec[63]]);
|
||||
let dc = id.unsigned_abs() as u8;
|
||||
(1..=5).contains(&dc).then_some(dc)
|
||||
}
|
||||
```
|
||||
|
||||
**3.** Открываем WebSocket к нужному DC с обязательным заголовком `Sec-WebSocket-Protocol: binary` и таймаутом 10 секунд:
|
||||
|
||||
```rust
|
||||
let (mut ws, _) = tokio::time::timeout(
|
||||
Duration::from_secs(10),
|
||||
tokio_tungstenite::connect_async_tls_with_config(req, None, false, Some(tls)),
|
||||
).await??;
|
||||
```
|
||||
|
||||
**4.** Отправляем буферизованные 64 байта init-пакета как первый WebSocket-фрейм. Дальше — двунаправленный relay в одном `tokio::select!` цикле:
|
||||
|
||||
```rust
|
||||
loop {
|
||||
tokio::select! {
|
||||
biased;
|
||||
msg = ws.next() => match msg {
|
||||
Some(Ok(Message::Binary(data))) => {
|
||||
tcp_w.write_all(data.as_ref()).await?;
|
||||
tcp_w.flush().await?;
|
||||
}
|
||||
Some(Ok(Message::Ping(p))) => {
|
||||
ws.send(Message::Pong(p)).await?;
|
||||
}
|
||||
_ => break,
|
||||
},
|
||||
n = tcp_r.read(&mut buf) => match n {
|
||||
Ok(0) | Err(_) => break,
|
||||
Ok(n) => { ws.send(Message::Binary(buf[..n].to_vec())).await?; }
|
||||
},
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
Ключевой момент — **Ping/Pong**. Без ответа на Ping сервер закрывает соединение через ~2 минуты. Первая версия это игнорировала — пользователи жаловались на обрывы.
|
||||
|
||||
### Не-Telegram трафик
|
||||
|
||||
Если destination IP не принадлежит Telegram — прямой TCP passthrough. Прокси не трогает ничего лишнего.
|
||||
|
||||
## Стабильность: что ломалось и как починили
|
||||
|
||||
**Проблема 1: Обрыв через 2 минуты.**
|
||||
WebSocket-сервер отправляет Ping-фреймы. Первая реализация использовала `split()` и два отдельных потока — Ping приходил в `read`-поток, а Pong нужно было отправить через `write`-поток. Решение: единый `tokio::select!` цикл без split. `biased` приоритизирует WS-чтение — Pong улетает мгновенно.
|
||||
|
||||
**Проблема 2: Неправильный DC.**
|
||||
IP-маппинг ненадёжен: в подсети 149.154.164-167 живут и DC2, и DC4. Если отправить данные не в тот DC — сервер дропает соединение. Решение: извлекать DC из obfuscated2 init через AES-256-CTR.
|
||||
|
||||
**Проблема 3: Зависание на подключении.**
|
||||
Если `kws*.web.telegram.org` не отвечает — прокси висел бесконечно. Решение: `tokio::time::timeout(10s)` на WebSocket connect.
|
||||
|
||||
**Проблема 4: Потеря данных.**
|
||||
TCP-write без `flush()` мог буферизовать данные. Telegram Desktop ожидал ответ, не получал его, переподключался. Решение: явный `flush()` после каждого write.
|
||||
|
||||
## UI: egui, не Electron
|
||||
|
||||
Нативный GUI через egui. Тёмная тема, минимальный интерфейс. Бинарник ~6 МБ, без зависимостей.
|
||||
|
||||
Одна кнопка — ПОДКЛЮЧИТЬ/ОТКЛЮЧИТЬ. Статистика в реальном времени: активные соединения, WebSocket-туннели, текущий DC, аптайм.
|
||||
|
||||
## Кроссплатформенность
|
||||
|
||||
Ни одной строки платформо-специфичного кода. Работает на:
|
||||
- **Windows** x64
|
||||
- **macOS** Intel + Apple Silicon
|
||||
- **Linux** x64
|
||||
|
||||
CI/CD через GitHub Actions — при создании тега автоматически собираются бинарники для всех платформ.
|
||||
|
||||
## Сравнение
|
||||
|
||||
| | GoodbyeDPI | Zapret | VPN | **TGLock** |
|
||||
|---|---|---|---|---|
|
||||
| Метод | Фрагментация | Desync | Туннель | WebSocket |
|
||||
| Обходит IP-шейпинг | Нет | Нет | Да | **Да** |
|
||||
| Нужен сервер | Нет | Нет | Да | **Нет** |
|
||||
| Весь трафик | Нет | Нет | Да | **Только Telegram** |
|
||||
| Кроссплатформа | Windows | Win/Mac/Linux | Да | **Win/Mac/Linux** |
|
||||
| Размер | ~1 МБ | ~2 МБ | Зависит | **~6 МБ** |
|
||||
|
||||
## Цифры
|
||||
|
||||
- **300** строк кода (proxy + UI)
|
||||
- **2** файла (`proxy.rs` + `main.rs`)
|
||||
- **3** платформы (Windows, macOS, Linux)
|
||||
- **0** серверов
|
||||
- **0₽**
|
||||
|
||||
## Скачать
|
||||
|
||||
**[github.com/by-sonic/tglock](https://github.com/by-sonic/tglock)** → Releases
|
||||
|
||||
Или собрать: `git clone ... && cargo build --release`
|
||||
|
||||
**P.S.** Для полного обхода блокировок (YouTube, Discord, Instagram) — **[by sonic VPN](https://t.me/bysonicvpn_bot)**.
|
||||
|
||||
---
|
||||
|
||||
*by sonic*
|
||||
|
||||
**Теги:** telegram, rust, websocket, socks5, mtproto, dpi, обход блокировок, open-source
|
||||
|
||||
**Хабы:** Rust · Open source · Сетевые технологии
|
||||
307
HABR_ARTICLE.md
307
HABR_ARTICLE.md
@@ -1,307 +0,0 @@
|
||||
# Как я написал обход блокировки Telegram на Rust — без VPN, без серверов, через WebSocket
|
||||
|
||||
**Простой · 7 мин · Rust · Open source · Windows · Сетевые технологии · Из песочницы**
|
||||
|
||||
**TL;DR:** Написал open-source десктопное приложение **TG Unblock** на Rust, которое в один клик обходит блокировку Telegram через локальный WebSocket-прокси. Трафик заворачивается в обычный HTTPS к `web.telegram.org` — DPI не видит MTProto, провайдер не может шейпить. Без VPN, без серверов, без абонентки. Код на GitHub — [by-sonic/tglock](https://github.com/by-sonic/tglock).
|
||||
|
||||
---
|
||||
|
||||
## Предыстория: почему GoodbyeDPI не спасает
|
||||
|
||||
С весны 2025 года Telegram в России стал работать, мягко говоря, через боль. Сообщения доходят по 10 секунд, медиа не грузятся, звонки рвутся. Классическая картина: провайдер + DPI = страдания.
|
||||
|
||||
Первое, что приходит в голову — **GoodbyeDPI**. Запустил, пакеты фрагментируются, DPI не узнаёт MTProto... и вроде работает. Но:
|
||||
|
||||
- **Пинг 200+ мс** — при норме 40–60
|
||||
- **Постоянные переподключения** — DPI переобучается и режет соединения
|
||||
- **IP-шейпинг** — провайдер троттлит весь трафик к подсетям Telegram (149.154.x.x, 91.108.x.x)
|
||||
|
||||
GoodbyeDPI обманывает DPI на уровне пакетов, но **не решает проблему IP-шейпинга**. Если провайдер тупо режет скорость ко всем IP Telegram — хоть как фрагментируй, будет медленно.
|
||||
|
||||
VPN — вариант. Но:
|
||||
- Платные стоят денег и сливают скорость
|
||||
- Бесплатные сливают данные
|
||||
- Не все работают стабильно
|
||||
- Для одного Telegram гонять весь трафик через VPN — оверкилл
|
||||
|
||||
Нужно решение, которое **маскирует сам факт подключения к Telegram**, а не просто прячет протокол.
|
||||
|
||||
## Идея: WebSocket-туннель через web.telegram.org
|
||||
|
||||
Я провёл серию тестов. Прямое подключение к серверам Telegram (149.154.167.51:443) — либо таймаут, либо 200+ мс. А вот `web.telegram.org` отвечает стабильно за 50–80 мс через HTTPS. Провайдер его не трогает — это же «обычный сайт».
|
||||
|
||||
И тут я полез в [документацию MTProto](https://core.telegram.org/mtproto/transports) и нашёл золотую жилу:
|
||||
|
||||
> **WebSocket:** Implementation of the WebSocket transport is pretty much the same as with TCP... all data received and sent through WebSocket messages is to be treated as a single duplex stream of bytes, just like with TCP.
|
||||
|
||||
Telegram **официально поддерживает WebSocket-транспорт**. Серверы `pluto.web.telegram.org`, `venus.web.telegram.org` и т.д. — это не просто веб-клиент. Это **полноценные точки входа в сеть Telegram** через WSS.
|
||||
|
||||
Схема:
|
||||
|
||||
```
|
||||
Telegram Desktop
|
||||
│
|
||||
▼ SOCKS5
|
||||
┌──────────────────┐
|
||||
│ TG Unblock │ 127.0.0.1:1080
|
||||
│ WS-прокси │
|
||||
└──────┬───────────┘
|
||||
│
|
||||
├── IP Telegram? ──► WSS к {dc}.web.telegram.org/apiws
|
||||
│ (провайдер видит: HTTPS к web.telegram.org)
|
||||
│
|
||||
└── Другой IP? ────► Прямой TCP (без изменений)
|
||||
```
|
||||
|
||||
Провайдер видит:
|
||||
- Соединение к `venus.web.telegram.org` по порту 443
|
||||
- Обычный TLS/HTTPS трафик
|
||||
- Никакого MTProto
|
||||
|
||||
DPI видит:
|
||||
- Ничего подозрительного
|
||||
- Обычный WebSocket внутри HTTPS
|
||||
|
||||
Результат:
|
||||
- **Полная скорость** — провайдер не шейпит web.telegram.org
|
||||
- **Нет переподключений** — DPI не трогает HTTPS
|
||||
- **Нулевая задержка** — нет промежуточных серверов, трафик идёт напрямую к Telegram
|
||||
|
||||
## Реализация: Rust, SOCKS5, WebSocket
|
||||
|
||||
### Почему Rust?
|
||||
|
||||
Не Electron. Не Python. Не Node.js. **Rust.** Потому что:
|
||||
- Один бинарник ~6 МБ, без зависимостей
|
||||
- Нативная скорость — прокси не должен добавлять задержку
|
||||
- Async I/O через tokio — тысячи одновременных соединений
|
||||
- Компилируется, запускается, работает
|
||||
|
||||
### Архитектура
|
||||
|
||||
Приложение состоит из 4 модулей:
|
||||
|
||||
| Модуль | Что делает |
|
||||
|---|---|
|
||||
| `main.rs` | GUI на egui + управление прокси |
|
||||
| `ws_proxy.rs` | SOCKS5-сервер + WebSocket-туннель |
|
||||
| `bypass.rs` | DNS-настройка, системные утилиты |
|
||||
| `network.rs` | Сетевая диагностика |
|
||||
|
||||
### SOCKS5 → WebSocket: как это работает
|
||||
|
||||
Когда Telegram Desktop подключается через SOCKS5-прокси, происходит следующее:
|
||||
|
||||
**1. SOCKS5 handshake**
|
||||
|
||||
```rust
|
||||
// Клиент: [0x05, 0x01, 0x00] — SOCKS5, 1 метод, no auth
|
||||
// Сервер: [0x05, 0x00] — принято
|
||||
// Клиент: [0x05, 0x01, 0x00, 0x01, IP, PORT] — CONNECT к IP:PORT
|
||||
```
|
||||
|
||||
**2. Определение DC по IP**
|
||||
|
||||
Telegram использует фиксированные подсети для каждого Data Center. Из [документации](https://core.telegram.org/mtproto/transports):
|
||||
|
||||
```rust
|
||||
fn telegram_dc(ip: Ipv4Addr) -> Option<u8> {
|
||||
let o = ip.octets();
|
||||
match (o[0], o[1]) {
|
||||
(149, 154) => Some(match o[2] {
|
||||
160..=163 => 1, // DC1
|
||||
164..=167 => 2, // DC2
|
||||
168..=171 => 3, // DC3
|
||||
172..=175 => 1, // DC1 alt
|
||||
_ => 2,
|
||||
}),
|
||||
(91, 108) => Some(match o[2] {
|
||||
56..=59 => 5, // DC5
|
||||
8..=11 => 3, // DC3
|
||||
12..=15 => 4, // DC4
|
||||
_ => 2,
|
||||
}),
|
||||
(91, 105) => Some(2),
|
||||
(185, 76) => Some(2),
|
||||
_ => None,
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
**3. WebSocket-туннель**
|
||||
|
||||
Каждый DC имеет именованный WebSocket-эндпоинт (имена из официальной документации Telegram):
|
||||
|
||||
| DC | Имя | URL |
|
||||
|---|---|---|
|
||||
| 1 | Pluto | `wss://pluto.web.telegram.org/apiws` |
|
||||
| 2 | Venus | `wss://venus.web.telegram.org/apiws` |
|
||||
| 3 | Aurora | `wss://aurora.web.telegram.org/apiws` |
|
||||
| 4 | Vesta | `wss://vesta.web.telegram.org/apiws` |
|
||||
| 5 | Flora | `wss://flora.web.telegram.org/apiws` |
|
||||
|
||||
Обязательный заголовок (из доки Telegram): `Sec-WebSocket-Protocol: binary`.
|
||||
|
||||
```rust
|
||||
let mut request = ws_url.as_str().into_client_request()?;
|
||||
request.headers_mut().insert(
|
||||
"Sec-WebSocket-Protocol", "binary".parse()?,
|
||||
);
|
||||
|
||||
let (ws, _) = tokio_tungstenite::connect_async_tls_with_config(
|
||||
request, None, false, Some(connector),
|
||||
).await?;
|
||||
```
|
||||
|
||||
**4. Двунаправленный relay**
|
||||
|
||||
Ключевая цитата из документации Telegram:
|
||||
|
||||
> All data received and sent through WebSocket messages is to be treated as a **single duplex stream of bytes**, just like with TCP.
|
||||
|
||||
Это значит, что нам не нужно парсить MTProto. Просто relay байтов: TCP → WebSocket binary frame, WebSocket binary frame → TCP.
|
||||
|
||||
```rust
|
||||
let up = async {
|
||||
let mut buf = vec![0u8; 32768];
|
||||
loop {
|
||||
match tcp_rx.read(&mut buf).await {
|
||||
Ok(0) => break,
|
||||
Ok(n) => {
|
||||
let msg = Message::Binary(buf[..n].to_vec());
|
||||
if ws_tx.send(msg).await.is_err() { break; }
|
||||
}
|
||||
Err(_) => break,
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
let down = async {
|
||||
while let Some(Ok(msg)) = ws_rx.next().await {
|
||||
if let Message::Binary(data) = msg {
|
||||
if tcp_tx.write_all(&data).await.is_err() { break; }
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
tokio::select! { _ = up => {}, _ = down => {} }
|
||||
```
|
||||
|
||||
### GUI: egui, не Electron
|
||||
|
||||
Нативный GUI через `egui` / `eframe`. Никакого браузера, никакого DOM, никакого JavaScript. Вся отрисовка — immediate mode, 60 FPS.
|
||||
|
||||
Кнопка «Запустить обход» делает:
|
||||
1. Меняет DNS на Cloudflare (1.1.1.1) — обходит DNS-блокировку
|
||||
2. Запускает SOCKS5-прокси на 127.0.0.1:1080
|
||||
3. Предлагает автонастройку Telegram через `tg://socks?server=127.0.0.1&port=1080`
|
||||
|
||||
Кнопка «Настроить автоматически» — открывает Telegram Desktop с готовой конфигурацией прокси. Один клик.
|
||||
|
||||
## Технические детали, которые пришлось решить
|
||||
|
||||
### Проблема 1: Не-Telegram трафик
|
||||
|
||||
Если Telegram Desktop пускает через SOCKS5 не только MTProto, но и запросы к CDN, стикер-серверам, обновлениям — их нельзя заворачивать в WebSocket. Решение: проверяем IP по маппингу Telegram-подсетей. Telegram IP → WebSocket. Всё остальное → прямой TCP passthrough.
|
||||
|
||||
### Проблема 2: Определение DC
|
||||
|
||||
Telegram Desktop использует obfuscated2 транспорт. Первые 64 байта — зашифрованный хендшейк, в котором закодирован DC ID. Парсить его — целый проект.
|
||||
|
||||
Решение проще: определяем DC по destination IP. Telegram использует фиксированные подсети для каждого DC — маппинг стабильный и документированный.
|
||||
|
||||
### Проблема 3: TLS к WebSocket-эндпоинтам
|
||||
|
||||
WebSocket-соединение идёт через WSS (TLS). Используем `native-tls` — системные сертификаты Windows, без привязки к OpenSSL.
|
||||
|
||||
```rust
|
||||
let connector = tokio_tungstenite::Connector::NativeTls(
|
||||
native_tls::TlsConnector::new()?,
|
||||
);
|
||||
```
|
||||
|
||||
### Проблема 4: Graceful shutdown
|
||||
|
||||
При остановке прокси нужно:
|
||||
- Сбросить DNS обратно на DHCP
|
||||
- Корректно закрыть все WebSocket-соединения
|
||||
- Не оставить Telegram без связи
|
||||
|
||||
Используем `AtomicBool` для флага остановки — все задачи проверяют его и завершаются.
|
||||
|
||||
## Сравнение с альтернативами
|
||||
|
||||
| | GoodbyeDPI | Zapret | VPN | **TG Unblock** |
|
||||
|---|---|---|---|---|
|
||||
| Подход | Фрагментация пакетов | Desync пакетов | Туннель через сервер | WebSocket-туннель |
|
||||
| DPI видит MTProto? | Нет | Нет | Нет | **Нет** |
|
||||
| IP-шейпинг? | Не обходит | Не обходит | Обходит | **Обходит** |
|
||||
| Нужен сервер? | Нет | Нет | Да | **Нет** |
|
||||
| Скорость | Зависит от DPI | Зависит от DPI | Зависит от сервера | **Полная** |
|
||||
| Весь трафик? | Нет | Нет | Да | **Только Telegram** |
|
||||
| Стоимость | Бесплатно | Бесплатно | $3–10/мес | **Бесплатно** |
|
||||
|
||||
## Стек
|
||||
|
||||
| Технология | Зачем |
|
||||
|---|---|
|
||||
| **Rust** | Скорость, один бинарник, без зависимостей |
|
||||
| **egui / eframe** | Нативный GUI без браузера |
|
||||
| **tokio** | Async I/O, тысячи соединений |
|
||||
| **tokio-tungstenite** | WebSocket-клиент с TLS |
|
||||
| **native-tls** | Системные сертификаты Windows |
|
||||
| **GitHub Actions** | CI/CD — автобилд при новом теге |
|
||||
|
||||
## Цифры
|
||||
|
||||
- **5 DC** — полный маппинг всех Telegram Data Center
|
||||
- **1 бинарник** — ~6 МБ, без зависимостей
|
||||
- **0 серверов** — всё работает локально
|
||||
- **0₽** — полностью бесплатно и open-source
|
||||
- **1 клик** — от запуска до работающего Telegram
|
||||
|
||||
## Как попробовать
|
||||
|
||||
### Скачать готовый .exe
|
||||
|
||||
1. Скачайте `tg_unblock.exe` из [Releases](https://github.com/by-sonic/tglock/releases)
|
||||
2. Запустите (желательно от администратора — для DNS)
|
||||
3. Нажмите **«Запустить обход»**
|
||||
4. Нажмите **«Настроить автоматически»**
|
||||
5. В Telegram нажмите «Подключить»
|
||||
|
||||
### Собрать из исходников
|
||||
|
||||
```bash
|
||||
git clone https://github.com/by-sonic/tglock.git
|
||||
cd tglock
|
||||
cargo build --release
|
||||
# Бинарник: target/release/tg_unblock.exe
|
||||
```
|
||||
|
||||
## Что дальше
|
||||
|
||||
- **Автоопределение DC из obfuscated2** — парсинг первых 64 байт для точного маппинга
|
||||
- **Fallback на GoodbyeDPI** — если WebSocket-эндпоинт недоступен
|
||||
- **Linux / macOS** — porability через tokio + egui (уже почти готово)
|
||||
- **Статистика** — скорость, задержка, количество туннелей в реальном времени
|
||||
|
||||
## Вместо заключения
|
||||
|
||||
Telegram — это не просто мессенджер. Для миллионов людей это рабочий инструмент, канал связи, источник информации. Когда он работает через боль — страдают все.
|
||||
|
||||
GoodbyeDPI — отличный инструмент, но у него есть потолок. Когда DPI побеждён, а трафик всё равно шейпится — нужен другой подход. WebSocket-туннель через `web.telegram.org` — это как проехать мимо камеры на легальной машине вместо того, чтобы заклеивать номера.
|
||||
|
||||
Код полностью открыт. Если пригодился — поставьте звезду на GitHub. Если нашли баг — PR приветствуются.
|
||||
|
||||
**GitHub:** [github.com/by-sonic/tglock](https://github.com/by-sonic/tglock)
|
||||
|
||||
**P.S.** Если нужен полный обход блокировок для всех приложений (YouTube, Discord, Instagram и др.) — попробуйте [by sonic VPN](https://t.me/bysonicvpn_bot). Быстрый, стабильный, без ограничений скорости.
|
||||
|
||||
---
|
||||
|
||||
*by sonic*
|
||||
|
||||
**Теги:** telegram, dpi bypass, websocket, rust, socks5, mtproto, обход блокировок, open-source
|
||||
|
||||
**Хабы:** Rust · Open source · Windows · Сетевые технологии
|
||||
191
README.md
191
README.md
@@ -1,40 +1,65 @@
|
||||
<p align="center">
|
||||
<h1 align="center">TG Unblock</h1>
|
||||
<h1 align="center">TGLock</h1>
|
||||
<p align="center"><b>Обход блокировки Telegram через WebSocket-туннель</b></p>
|
||||
<p align="center">Без VPN. Без серверов. Без абонентки. Один клик.</p>
|
||||
<p align="center">
|
||||
<b>Обход блокировки Telegram через WebSocket-туннель</b><br>
|
||||
Без VPN. Без серверов. Без абонентки. Один клик.
|
||||
</p>
|
||||
<p align="center">
|
||||
<a href="https://github.com/by-sonic/tglock/releases"><img src="https://img.shields.io/github/v/release/by-sonic/tglock?style=for-the-badge&color=blue" alt="Release"></a>
|
||||
<a href="https://github.com/by-sonic/tglock/blob/main/LICENSE"><img src="https://img.shields.io/github/license/by-sonic/tglock?style=for-the-badge" alt="License"></a>
|
||||
<a href="https://github.com/by-sonic/tglock/stargazers"><img src="https://img.shields.io/github/stars/by-sonic/tglock?style=for-the-badge&color=yellow" alt="Stars"></a>
|
||||
<img src="https://img.shields.io/badge/rust-1.70%2B-orange?style=for-the-badge&logo=rust" alt="Rust">
|
||||
<img src="https://img.shields.io/badge/platform-Windows-0078D6?style=for-the-badge&logo=windows" alt="Windows">
|
||||
<a href="https://github.com/by-sonic/tglock/releases/latest"><img src="https://img.shields.io/github/v/release/by-sonic/tglock?style=flat-square&color=blue" alt="Release"></a>
|
||||
<img src="https://img.shields.io/badge/platform-Windows%20%7C%20macOS%20%7C%20Linux-333?style=flat-square" alt="Platform">
|
||||
<img src="https://img.shields.io/badge/rust-stable-orange?style=flat-square&logo=rust" alt="Rust">
|
||||
<a href="LICENSE"><img src="https://img.shields.io/github/license/by-sonic/tglock?style=flat-square" alt="MIT"></a>
|
||||
</p>
|
||||
</p>
|
||||
|
||||
---
|
||||
|
||||
## Что это?
|
||||
|
||||
**TG Unblock** — десктопное приложение на Rust, которое обходит блокировку Telegram через локальный WebSocket-прокси. Провайдер видит обычный HTTPS к `web.telegram.org`, а не MTProto — DPI не может обнаружить и заблокировать трафик.
|
||||
|
||||
### Почему не GoodbyeDPI / Zapret?
|
||||
|
||||
| | GoodbyeDPI | Zapret | **TG Unblock** |
|
||||
|---|---|---|---|
|
||||
| Метод | Фрагментация пакетов | Desync пакетов | WebSocket-туннель |
|
||||
| DPI видит MTProto? | Нет (обфускация) | Нет (desync) | **Нет (обычный HTTPS)** |
|
||||
| IP-шейпинг обходит? | Нет | Нет | **Да** |
|
||||
| Скорость | Зависит от DPI | Зависит от DPI | **Полная** |
|
||||
| Переподключения | Возможны | Возможны | **Нет** |
|
||||
| Настройка | Много параметров | Стратегии | **Один клик** |
|
||||
|
||||
## Скачать
|
||||
|
||||
> **[Скачать последний релиз](https://github.com/by-sonic/tglock/releases)**
|
||||
**[Последний релиз](https://github.com/by-sonic/tglock/releases/latest)**
|
||||
|
||||
Или собрать из исходников:
|
||||
| Файл | Платформа |
|
||||
|---|---|
|
||||
| `tglock.exe` | Windows x64 |
|
||||
| `tglock-macos-arm64` | macOS Apple Silicon (M1–M4) |
|
||||
| `tglock-macos-x64` | macOS Intel |
|
||||
| `tglock-linux-x64` | Linux x64 |
|
||||
|
||||
## Как пользоваться
|
||||
|
||||
1. Скачай и запусти
|
||||
2. Нажми **ПОДКЛЮЧИТЬ**
|
||||
3. Нажми **Настроить автоматически** → в Telegram нажми «Подключить»
|
||||
4. Готово
|
||||
|
||||
Ручная настройка: Telegram → Настройки → Продвинутые → Тип соединения → SOCKS5 → `127.0.0.1:1080`
|
||||
|
||||
## Как это работает
|
||||
|
||||
```
|
||||
Telegram Desktop → SOCKS5 (127.0.0.1:1080) → TGLock → WSS (web.telegram.org) → DC
|
||||
```
|
||||
|
||||
1. Локальный SOCKS5-прокси перехватывает соединения Telegram
|
||||
2. Из MTProto init-пакета извлекается номер DC (AES-256-CTR)
|
||||
3. Трафик заворачивается в WebSocket через `kws{dc}.web.telegram.org`
|
||||
4. Провайдер видит обычный HTTPS к `web.telegram.org`
|
||||
5. Остальной трафик проходит напрямую
|
||||
|
||||
## Почему не GoodbyeDPI / Zapret?
|
||||
|
||||
GoodbyeDPI и Zapret фрагментируют пакеты чтобы обмануть DPI. Но если провайдер **шейпит по IP** — они бесполезны.
|
||||
|
||||
TGLock маскирует трафик под обычный HTTPS. DPI не видит MTProto. IP-шейпинг не работает — `web.telegram.org` не блокируется.
|
||||
|
||||
## Стек
|
||||
|
||||
| | |
|
||||
|---|---|
|
||||
| Rust | Один бинарник, нативная скорость |
|
||||
| egui | GUI без браузера и Electron |
|
||||
| tokio | Async I/O |
|
||||
| tokio-tungstenite | WebSocket + TLS |
|
||||
|
||||
## Сборка
|
||||
|
||||
```bash
|
||||
git clone https://github.com/by-sonic/tglock.git
|
||||
@@ -42,118 +67,14 @@ cd tglock
|
||||
cargo build --release
|
||||
```
|
||||
|
||||
Готовый `.exe` будет в `target/release/tg_unblock.exe`.
|
||||
## VPN
|
||||
|
||||
## Как пользоваться
|
||||
|
||||
1. Запустите `tg_unblock.exe`
|
||||
2. Нажмите **"Запустить обход"**
|
||||
3. Нажмите **"Настроить автоматически"** — откроется Telegram, нажмите "Подключить"
|
||||
4. Готово. Telegram работает на полной скорости.
|
||||
|
||||
### Ручная настройка прокси
|
||||
|
||||
Если автонастройка не сработала:
|
||||
|
||||
**Telegram Desktop** → Настройки → Продвинутые → Тип соединения → **Использовать SOCKS5-прокси**
|
||||
|
||||
| Параметр | Значение |
|
||||
|---|---|
|
||||
| Сервер | `127.0.0.1` |
|
||||
| Порт | `1080` |
|
||||
| Логин | *пусто* |
|
||||
| Пароль | *пусто* |
|
||||
|
||||
## Как это работает
|
||||
|
||||
```
|
||||
Telegram Desktop
|
||||
│
|
||||
▼ (SOCKS5)
|
||||
┌──────────────────┐
|
||||
│ TG Unblock │ 127.0.0.1:1080
|
||||
│ WS-прокси │
|
||||
└──────┬───────────┘
|
||||
│
|
||||
▼ (определяет DC по IP)
|
||||
│
|
||||
├── Telegram IP? ──► WSS-туннель к {dc}.web.telegram.org/apiws
|
||||
│ (провайдер видит обычный HTTPS)
|
||||
│
|
||||
└── Другой IP? ────► Прямое TCP-соединение (без изменений)
|
||||
```
|
||||
|
||||
### DC-маппинг
|
||||
|
||||
Приложение автоматически определяет Data Center по IP-адресу и маршрутизирует через правильный WebSocket-эндпоинт:
|
||||
|
||||
| DC | Подсеть | WebSocket |
|
||||
|---|---|---|
|
||||
| DC1 | `149.154.160.0/22` | `wss://pluto.web.telegram.org/apiws` |
|
||||
| DC2 | `149.154.164.0/22` | `wss://venus.web.telegram.org/apiws` |
|
||||
| DC3 | `149.154.168.0/22` | `wss://aurora.web.telegram.org/apiws` |
|
||||
| DC4 | `91.108.12.0/22` | `wss://vesta.web.telegram.org/apiws` |
|
||||
| DC5 | `91.108.56.0/22` | `wss://flora.web.telegram.org/apiws` |
|
||||
|
||||
Имена DC (`pluto`, `venus`, `aurora`, `vesta`, `flora`) — из [официальной документации MTProto](https://core.telegram.org/mtproto/transports).
|
||||
|
||||
## Стек
|
||||
|
||||
| Что | Зачем |
|
||||
|---|---|
|
||||
| **Rust** | Скорость, безопасность, один бинарник без зависимостей |
|
||||
| **egui / eframe** | Нативный GUI без Electron, без браузера |
|
||||
| **tokio** | Async I/O для высокопроизводительного проксирования |
|
||||
| **tokio-tungstenite** | WebSocket-клиент с TLS |
|
||||
| **native-tls** | TLS через системные сертификаты Windows |
|
||||
|
||||
## Структура проекта
|
||||
|
||||
```
|
||||
tglock/
|
||||
├── Cargo.toml # Зависимости
|
||||
├── src/
|
||||
│ ├── main.rs # GUI + управление прокси
|
||||
│ ├── ws_proxy.rs # SOCKS5-сервер + WebSocket-туннель
|
||||
│ ├── bypass.rs # DNS-настройка, утилиты Windows
|
||||
│ └── network.rs # Сетевая диагностика
|
||||
└── tg_blacklist.txt # IP-подсети и домены Telegram
|
||||
```
|
||||
|
||||
## Требования
|
||||
|
||||
- Windows 10/11
|
||||
- [Rust 1.70+](https://rustup.rs/) (для сборки из исходников)
|
||||
- Права администратора (для смены DNS, опционально)
|
||||
|
||||
## FAQ
|
||||
|
||||
**Q: Это VPN?**
|
||||
A: Нет. Трафик не идёт через сторонние серверы. Прокси работает локально и туннелирует только Telegram-трафик через WebSocket к официальным серверам Telegram.
|
||||
|
||||
**Q: Это безопасно?**
|
||||
A: Весь код открыт. Никакой телеметрии. Никаких данных не отправляется. Соединение с Telegram остаётся end-to-end зашифрованным (MTProto).
|
||||
|
||||
**Q: Будет ли работать с мобильным Telegram?**
|
||||
A: Пока только Telegram Desktop. Для мобильных устройств рекомендуем [by sonic VPN](https://t.me/bysonicvpn_bot).
|
||||
|
||||
**Q: Замедляется ли интернет?**
|
||||
A: Нет. Проксируется только трафик к серверам Telegram. Весь остальной трафик идёт напрямую.
|
||||
|
||||
## VPN для полного обхода
|
||||
|
||||
Если нужен обход блокировок для **всех** приложений (YouTube, Discord, Instagram и др.) — попробуйте **[by sonic VPN](https://t.me/bysonicvpn_bot)**. Быстрый, без ограничений скорости.
|
||||
Для обхода блокировок **всех** приложений — **[by sonic VPN](https://t.me/bysonicvpn_bot)**
|
||||
|
||||
## Лицензия
|
||||
|
||||
MIT — делайте что хотите.
|
||||
|
||||
## Автор
|
||||
|
||||
**by sonic** — [@bysonicvpn_bot](https://t.me/bysonicvpn_bot)
|
||||
MIT
|
||||
|
||||
---
|
||||
|
||||
<p align="center">
|
||||
<b>Если пригодилось — поставьте ⭐ на GitHub</b>
|
||||
</p>
|
||||
<p align="center"><b>by sonic</b></p>
|
||||
|
||||
201
src/bypass.rs
201
src/bypass.rs
@@ -1,201 +0,0 @@
|
||||
use std::path::{Path, PathBuf};
|
||||
use std::process::Command;
|
||||
|
||||
pub fn check_admin() -> bool {
|
||||
let output = Command::new("net")
|
||||
.args(["session"])
|
||||
.output();
|
||||
matches!(output, Ok(o) if o.status.success())
|
||||
}
|
||||
|
||||
pub fn set_dns(adapter: &str, primary: &str, secondary: &str) -> Result<(), String> {
|
||||
let out1 = Command::new("netsh")
|
||||
.args([
|
||||
"interface", "ipv4", "set", "dnsservers",
|
||||
adapter, "static", primary, "primary", "validate=no",
|
||||
])
|
||||
.output()
|
||||
.map_err(|e| format!("netsh error: {}", e))?;
|
||||
|
||||
if !out1.status.success() {
|
||||
let stderr = String::from_utf8_lossy(&out1.stderr);
|
||||
return Err(format!("Failed to set primary DNS: {}", stderr));
|
||||
}
|
||||
|
||||
let out2 = Command::new("netsh")
|
||||
.args([
|
||||
"interface", "ipv4", "add", "dnsservers",
|
||||
adapter, secondary, "index=2", "validate=no",
|
||||
])
|
||||
.output()
|
||||
.map_err(|e| format!("netsh error: {}", e))?;
|
||||
|
||||
if !out2.status.success() {
|
||||
// Non-critical: secondary DNS may already exist
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn reset_dns(adapter: &str) -> Result<(), String> {
|
||||
let out = Command::new("netsh")
|
||||
.args([
|
||||
"interface", "ipv4", "set", "dnsservers",
|
||||
adapter, "dhcp",
|
||||
])
|
||||
.output()
|
||||
.map_err(|e| format!("netsh error: {}", e))?;
|
||||
|
||||
if !out.status.success() {
|
||||
let stderr = String::from_utf8_lossy(&out.stderr);
|
||||
return Err(format!("Failed to reset DNS: {}", stderr));
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn flush_dns() {
|
||||
let _ = Command::new("ipconfig")
|
||||
.args(["/flushdns"])
|
||||
.output();
|
||||
}
|
||||
|
||||
pub fn find_goodbyedpi() -> Option<String> {
|
||||
let exe_dir = std::env::current_exe()
|
||||
.ok()
|
||||
.and_then(|p| p.parent().map(|p| p.to_path_buf()))
|
||||
.unwrap_or_else(|| PathBuf::from("."));
|
||||
|
||||
let search_dirs = vec![
|
||||
exe_dir.join("tools"),
|
||||
exe_dir.join("tools").join("goodbyedpi"),
|
||||
exe_dir.clone(),
|
||||
PathBuf::from("tools"),
|
||||
PathBuf::from("tools").join("goodbyedpi"),
|
||||
PathBuf::from("."),
|
||||
];
|
||||
|
||||
for dir in &search_dirs {
|
||||
// Check common locations
|
||||
for sub in &["x86_64", "x86", ""] {
|
||||
let candidate = if sub.is_empty() {
|
||||
dir.join("goodbyedpi.exe")
|
||||
} else {
|
||||
dir.join(sub).join("goodbyedpi.exe")
|
||||
};
|
||||
if candidate.exists() {
|
||||
return Some(candidate.to_string_lossy().to_string());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Recursive search in tools/
|
||||
if let Ok(entries) = find_file_recursive(Path::new("tools"), "goodbyedpi.exe") {
|
||||
if !entries.is_empty() {
|
||||
return Some(entries[0].to_string_lossy().to_string());
|
||||
}
|
||||
}
|
||||
|
||||
None
|
||||
}
|
||||
|
||||
fn find_file_recursive(dir: &Path, filename: &str) -> Result<Vec<PathBuf>, std::io::Error> {
|
||||
let mut results = Vec::new();
|
||||
if !dir.exists() {
|
||||
return Ok(results);
|
||||
}
|
||||
for entry in std::fs::read_dir(dir)? {
|
||||
let entry = entry?;
|
||||
let path = entry.path();
|
||||
if path.is_file() && path.file_name().map(|n| n == filename).unwrap_or(false) {
|
||||
results.push(path);
|
||||
} else if path.is_dir() {
|
||||
results.extend(find_file_recursive(&path, filename)?);
|
||||
}
|
||||
}
|
||||
Ok(results)
|
||||
}
|
||||
|
||||
pub fn get_blacklist_path() -> Option<String> {
|
||||
let candidates = vec![
|
||||
PathBuf::from("tg_blacklist.txt"),
|
||||
PathBuf::from("tools").join("tg_blacklist.txt"),
|
||||
std::env::current_exe()
|
||||
.ok()
|
||||
.and_then(|p| p.parent().map(|p| p.join("tg_blacklist.txt")))
|
||||
.unwrap_or_default(),
|
||||
];
|
||||
|
||||
for path in candidates {
|
||||
if path.exists() {
|
||||
return Some(path.to_string_lossy().to_string());
|
||||
}
|
||||
}
|
||||
None
|
||||
}
|
||||
|
||||
pub fn start_goodbyedpi(exe_path: &str, args: &[&str], blacklist: Option<&str>) -> Result<(), String> {
|
||||
let mut cmd = Command::new(exe_path);
|
||||
cmd.args(args);
|
||||
|
||||
if let Some(bl) = blacklist {
|
||||
cmd.args(["--blacklist", bl]);
|
||||
}
|
||||
|
||||
cmd.spawn().map_err(|e| format!("Failed to start GoodbyeDPI: {}", e))?;
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn kill_goodbyedpi() {
|
||||
let _ = Command::new("taskkill")
|
||||
.args(["/f", "/im", "goodbyedpi.exe"])
|
||||
.output();
|
||||
}
|
||||
|
||||
pub fn download_goodbyedpi() -> Result<String, String> {
|
||||
let tools_dir = PathBuf::from("tools");
|
||||
std::fs::create_dir_all(&tools_dir)
|
||||
.map_err(|e| format!("Cannot create tools dir: {}", e))?;
|
||||
|
||||
let zip_path = tools_dir.join("goodbyedpi.zip");
|
||||
let url = "https://github.com/ValdikSS/GoodbyeDPI/releases/download/0.2.3rc3/goodbyedpi-0.2.3rc3-2.zip";
|
||||
|
||||
// Download using powershell
|
||||
let dl_script = format!(
|
||||
"[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; Invoke-WebRequest -Uri '{}' -OutFile '{}' -UseBasicParsing",
|
||||
url,
|
||||
zip_path.to_string_lossy()
|
||||
);
|
||||
|
||||
let output = Command::new("powershell")
|
||||
.args(["-Command", &dl_script])
|
||||
.output()
|
||||
.map_err(|e| format!("Download failed: {}", e))?;
|
||||
|
||||
if !output.status.success() {
|
||||
let stderr = String::from_utf8_lossy(&output.stderr);
|
||||
return Err(format!("Download failed: {}", stderr));
|
||||
}
|
||||
|
||||
// Extract
|
||||
let extract_script = format!(
|
||||
"Expand-Archive -Path '{}' -DestinationPath '{}' -Force",
|
||||
zip_path.to_string_lossy(),
|
||||
tools_dir.to_string_lossy()
|
||||
);
|
||||
|
||||
let output = Command::new("powershell")
|
||||
.args(["-Command", &extract_script])
|
||||
.output()
|
||||
.map_err(|e| format!("Extraction failed: {}", e))?;
|
||||
|
||||
if !output.status.success() {
|
||||
let stderr = String::from_utf8_lossy(&output.stderr);
|
||||
return Err(format!("Extraction failed: {}", stderr));
|
||||
}
|
||||
|
||||
// Clean up zip
|
||||
let _ = std::fs::remove_file(&zip_path);
|
||||
|
||||
// Find the exe
|
||||
find_goodbyedpi().ok_or_else(|| "goodbyedpi.exe not found after extraction".to_string())
|
||||
}
|
||||
557
src/main.rs
557
src/main.rs
@@ -1,356 +1,365 @@
|
||||
#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")]
|
||||
|
||||
mod bypass;
|
||||
mod network;
|
||||
mod ws_proxy;
|
||||
mod proxy;
|
||||
|
||||
use std::sync::atomic::Ordering;
|
||||
use std::sync::{Arc, Mutex};
|
||||
use std::time::Instant;
|
||||
|
||||
use eframe::egui;
|
||||
|
||||
const PROXY_PORT: u16 = 1080;
|
||||
// -- Colors (GitHub Dark inspired) ------------------------------------------
|
||||
|
||||
const BG: egui::Color32 = egui::Color32::from_rgb(13, 17, 23);
|
||||
const SURFACE: egui::Color32 = egui::Color32::from_rgb(22, 27, 34);
|
||||
const BORDER: egui::Color32 = egui::Color32::from_rgb(48, 54, 61);
|
||||
const ACCENT: egui::Color32 = egui::Color32::from_rgb(88, 166, 255);
|
||||
const GREEN: egui::Color32 = egui::Color32::from_rgb(63, 185, 80);
|
||||
const RED: egui::Color32 = egui::Color32::from_rgb(248, 81, 73);
|
||||
const TEXT: egui::Color32 = egui::Color32::from_rgb(230, 237, 243);
|
||||
const TEXT2: egui::Color32 = egui::Color32::from_rgb(139, 148, 158);
|
||||
const AD_BG: egui::Color32 = egui::Color32::from_rgb(17, 21, 28);
|
||||
|
||||
fn main() -> eframe::Result<()> {
|
||||
let options = eframe::NativeOptions {
|
||||
viewport: egui::ViewportBuilder::default()
|
||||
.with_inner_size([680.0, 560.0])
|
||||
.with_min_inner_size([580.0, 460.0])
|
||||
.with_title("TG Unblock"),
|
||||
..Default::default()
|
||||
};
|
||||
|
||||
eframe::run_native(
|
||||
"TG Unblock",
|
||||
options,
|
||||
"TGLock",
|
||||
eframe::NativeOptions {
|
||||
viewport: egui::ViewportBuilder::default()
|
||||
.with_inner_size([520.0, 620.0])
|
||||
.with_min_inner_size([420.0, 500.0])
|
||||
.with_title("TGLock"),
|
||||
..Default::default()
|
||||
},
|
||||
Box::new(|cc| {
|
||||
setup_fonts(&cc.egui_ctx);
|
||||
apply_theme(&cc.egui_ctx);
|
||||
Ok(Box::new(App::new()))
|
||||
}),
|
||||
)
|
||||
}
|
||||
|
||||
fn setup_fonts(ctx: &egui::Context) {
|
||||
let mut fonts = egui::FontDefinitions::default();
|
||||
fonts.font_data.insert(
|
||||
"system".to_owned(),
|
||||
std::sync::Arc::new(egui::FontData::from_static(include_bytes!(
|
||||
"C:\\Windows\\Fonts\\segoeui.ttf"
|
||||
))),
|
||||
);
|
||||
fonts
|
||||
.families
|
||||
.entry(egui::FontFamily::Proportional)
|
||||
.or_default()
|
||||
.insert(0, "system".to_owned());
|
||||
fonts
|
||||
.families
|
||||
.entry(egui::FontFamily::Monospace)
|
||||
.or_default()
|
||||
.insert(0, "system".to_owned());
|
||||
ctx.set_fonts(fonts);
|
||||
fn apply_theme(ctx: &egui::Context) {
|
||||
let mut v = egui::Visuals::dark();
|
||||
v.panel_fill = BG;
|
||||
v.window_fill = SURFACE;
|
||||
v.extreme_bg_color = BG;
|
||||
v.faint_bg_color = SURFACE;
|
||||
v.override_text_color = Some(TEXT);
|
||||
|
||||
v.widgets.noninteractive.bg_fill = SURFACE;
|
||||
v.widgets.noninteractive.fg_stroke = egui::Stroke::new(1.0, TEXT2);
|
||||
v.widgets.noninteractive.bg_stroke = egui::Stroke::new(1.0, BORDER);
|
||||
|
||||
v.widgets.inactive.bg_fill = egui::Color32::from_rgb(33, 38, 45);
|
||||
v.widgets.inactive.fg_stroke = egui::Stroke::new(1.0, TEXT);
|
||||
v.widgets.inactive.bg_stroke = egui::Stroke::new(1.0, BORDER);
|
||||
|
||||
v.widgets.hovered.bg_fill = egui::Color32::from_rgb(48, 54, 61);
|
||||
v.widgets.hovered.fg_stroke = egui::Stroke::new(1.0, TEXT);
|
||||
|
||||
v.widgets.active.bg_fill = ACCENT;
|
||||
v.widgets.active.fg_stroke = egui::Stroke::new(1.0, BG);
|
||||
|
||||
ctx.set_visuals(v);
|
||||
}
|
||||
|
||||
// -- Log --------------------------------------------------------------------
|
||||
|
||||
#[derive(Clone)]
|
||||
struct LogEntry {
|
||||
text: String,
|
||||
is_error: bool,
|
||||
struct LogLine {
|
||||
ts: String,
|
||||
msg: String,
|
||||
err: bool,
|
||||
}
|
||||
|
||||
fn now_ts() -> String {
|
||||
let s = std::time::SystemTime::now()
|
||||
.duration_since(std::time::UNIX_EPOCH)
|
||||
.unwrap_or_default()
|
||||
.as_secs();
|
||||
format!("{:02}:{:02}:{:02}", (s / 3600) % 24, (s / 60) % 60, s % 60)
|
||||
}
|
||||
|
||||
fn log(log: &Arc<Mutex<Vec<LogLine>>>, msg: &str, err: bool) {
|
||||
log.lock().unwrap().push(LogLine {
|
||||
ts: now_ts(),
|
||||
msg: msg.into(),
|
||||
err,
|
||||
});
|
||||
}
|
||||
|
||||
// -- App --------------------------------------------------------------------
|
||||
|
||||
struct App {
|
||||
log: Arc<Mutex<Vec<LogEntry>>>,
|
||||
proxy_stats: Arc<ws_proxy::ProxyStats>,
|
||||
is_admin: bool,
|
||||
adapter_name: Arc<Mutex<Option<String>>>,
|
||||
dns_set: Arc<Mutex<bool>>,
|
||||
stats: Arc<proxy::Stats>,
|
||||
log: Arc<Mutex<Vec<LogLine>>>,
|
||||
started_at: Option<Instant>,
|
||||
lan_mode: bool,
|
||||
}
|
||||
|
||||
impl App {
|
||||
fn new() -> Self {
|
||||
let is_admin = bypass::check_admin();
|
||||
let app = Self {
|
||||
Self {
|
||||
stats: proxy::Stats::new(),
|
||||
log: Arc::new(Mutex::new(Vec::new())),
|
||||
proxy_stats: ws_proxy::ProxyStats::new(),
|
||||
is_admin,
|
||||
adapter_name: Arc::new(Mutex::new(None)),
|
||||
dns_set: Arc::new(Mutex::new(false)),
|
||||
};
|
||||
log_msg(&app.log, "Запущено", false);
|
||||
if !is_admin {
|
||||
log_msg(&app.log, "Нет прав администратора — DNS менять не получится", true);
|
||||
started_at: None,
|
||||
lan_mode: false,
|
||||
}
|
||||
{
|
||||
let adapter = app.adapter_name.clone();
|
||||
let log = app.log.clone();
|
||||
std::thread::spawn(move || {
|
||||
if let Some(name) = network::detect_adapter() {
|
||||
log_msg(&log, &format!("Адаптер: {}", name), false);
|
||||
*adapter.lock().unwrap() = Some(name);
|
||||
}
|
||||
});
|
||||
}
|
||||
app
|
||||
}
|
||||
|
||||
fn proxy_running(&self) -> bool {
|
||||
self.proxy_stats.running.load(Ordering::SeqCst)
|
||||
fn running(&self) -> bool {
|
||||
self.stats.running.load(Ordering::SeqCst)
|
||||
}
|
||||
|
||||
fn start_proxy(&self) {
|
||||
if self.proxy_running() {
|
||||
return;
|
||||
}
|
||||
let stats = self.proxy_stats.clone();
|
||||
let log = self.log.clone();
|
||||
let adapter = self.adapter_name.clone();
|
||||
let dns_set = self.dns_set.clone();
|
||||
let is_admin = self.is_admin;
|
||||
fn start(&mut self) {
|
||||
if self.running() { return; }
|
||||
self.started_at = Some(Instant::now());
|
||||
let stats = self.stats.clone();
|
||||
let lg = self.log.clone();
|
||||
let lan = self.lan_mode;
|
||||
|
||||
log(&lg, "Запускаю прокси...", false);
|
||||
|
||||
std::thread::spawn(move || {
|
||||
// DNS
|
||||
if is_admin {
|
||||
let aname = adapter.lock().unwrap().clone().or_else(network::detect_adapter);
|
||||
if let Some(ref name) = aname {
|
||||
if bypass::set_dns(name, "1.1.1.1", "1.0.0.1").is_ok() {
|
||||
bypass::flush_dns();
|
||||
log_msg(&log, "DNS → Cloudflare 1.1.1.1", false);
|
||||
*dns_set.lock().unwrap() = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
log_msg(&log, &format!("Запускаю WS-прокси на 127.0.0.1:{}...", PROXY_PORT), false);
|
||||
|
||||
let rt = tokio::runtime::Runtime::new().unwrap();
|
||||
let result = rt.block_on(ws_proxy::run_proxy(PROXY_PORT, stats));
|
||||
if let Err(e) = result {
|
||||
log_msg(&log, &format!("Прокси остановлен: {}", e), true);
|
||||
let r = rt.block_on(proxy::run(stats, lan));
|
||||
if let Err(e) = r {
|
||||
log(&lg, &format!("Ошибка: {}", e), true);
|
||||
}
|
||||
});
|
||||
|
||||
std::thread::sleep(std::time::Duration::from_millis(300));
|
||||
if self.proxy_running() {
|
||||
log_msg(&self.log, "Прокси запущен! Настройте Telegram.", false);
|
||||
std::thread::sleep(std::time::Duration::from_millis(250));
|
||||
if self.running() {
|
||||
let addr = if lan { "0.0.0.0" } else { "127.0.0.1" };
|
||||
log(&self.log, &format!("SOCKS5 на {}:{}", addr, proxy::PORT), false);
|
||||
if lan {
|
||||
log(&self.log, "LAN-режим: другие устройства могут подключаться", false);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn stop_proxy(&self) {
|
||||
self.proxy_stats.running.store(false, Ordering::SeqCst);
|
||||
log_msg(&self.log, "Прокси остановлен", false);
|
||||
fn stop(&mut self) {
|
||||
self.stats.running.store(false, Ordering::SeqCst);
|
||||
self.started_at = None;
|
||||
log(&self.log, "Остановлен", false);
|
||||
}
|
||||
|
||||
if *self.dns_set.lock().unwrap() {
|
||||
let adapter = self.adapter_name.clone();
|
||||
let log = self.log.clone();
|
||||
let dns_set = self.dns_set.clone();
|
||||
std::thread::spawn(move || {
|
||||
let aname = adapter.lock().unwrap().clone().or_else(network::detect_adapter);
|
||||
if let Some(ref name) = aname {
|
||||
let _ = bypass::reset_dns(name);
|
||||
bypass::flush_dns();
|
||||
*dns_set.lock().unwrap() = false;
|
||||
log_msg(&log, "DNS сброшен", false);
|
||||
}
|
||||
});
|
||||
fn uptime_str(&self) -> String {
|
||||
match self.started_at {
|
||||
Some(t) => {
|
||||
let s = t.elapsed().as_secs();
|
||||
format!("{:02}:{:02}:{:02}", s / 3600, (s / 60) % 60, s % 60)
|
||||
}
|
||||
None => "--:--:--".into(),
|
||||
}
|
||||
}
|
||||
|
||||
fn open_tg_proxy_link(&self) {
|
||||
let url = format!("tg://socks?server=127.0.0.1&port={}", PROXY_PORT);
|
||||
log_msg(&self.log, "Открываю настройку прокси в Telegram...", false);
|
||||
let _ = open::that(&url);
|
||||
}
|
||||
}
|
||||
|
||||
fn log_msg(log: &Arc<Mutex<Vec<LogEntry>>>, text: &str, err: bool) {
|
||||
let now = std::time::SystemTime::now()
|
||||
.duration_since(std::time::UNIX_EPOCH)
|
||||
.unwrap()
|
||||
.as_secs();
|
||||
let ts = format!("{:02}:{:02}:{:02}", (now % 86400) / 3600, (now % 3600) / 60, now % 60);
|
||||
log.lock().unwrap().push(LogEntry {
|
||||
text: text.to_string(),
|
||||
is_error: err,
|
||||
ts,
|
||||
});
|
||||
}
|
||||
|
||||
impl eframe::App for App {
|
||||
fn update(&mut self, ctx: &egui::Context, _frame: &mut eframe::Frame) {
|
||||
ctx.request_repaint_after(std::time::Duration::from_millis(400));
|
||||
ctx.request_repaint_after(std::time::Duration::from_millis(300));
|
||||
|
||||
let running = self.proxy_running();
|
||||
let active = self.proxy_stats.active_conn.load(Ordering::Relaxed);
|
||||
let total = self.proxy_stats.total_conn.load(Ordering::Relaxed);
|
||||
let ws = self.proxy_stats.ws_active.load(Ordering::Relaxed);
|
||||
let on = self.running();
|
||||
let active = self.stats.active.load(Ordering::Relaxed);
|
||||
let total = self.stats.total.load(Ordering::Relaxed);
|
||||
let ws = self.stats.ws.load(Ordering::Relaxed);
|
||||
let dc = self.stats.last_dc.load(Ordering::Relaxed);
|
||||
|
||||
// --- Top bar ---
|
||||
egui::TopBottomPanel::top("top").show(ctx, |ui| {
|
||||
ui.horizontal(|ui| {
|
||||
ui.heading("TG Unblock");
|
||||
ui.separator();
|
||||
if running {
|
||||
ui.colored_label(
|
||||
egui::Color32::from_rgb(80, 220, 120),
|
||||
egui::RichText::new("ПРОКСИ РАБОТАЕТ").strong(),
|
||||
);
|
||||
ui.separator();
|
||||
ui.label(format!("Соединений: {} (WS: {}) | Всего: {}", active, ws, total));
|
||||
} else {
|
||||
ui.label("Прокси не запущен");
|
||||
}
|
||||
});
|
||||
// === Ad bar (top) ===
|
||||
egui::TopBottomPanel::top("ad").show(ctx, |ui| {
|
||||
egui::Frame::new()
|
||||
.fill(AD_BG)
|
||||
.inner_margin(egui::Margin::symmetric(12, 6))
|
||||
.show(ui, |ui| {
|
||||
ui.horizontal(|ui| {
|
||||
ui.colored_label(ACCENT, egui::RichText::new("by sonic VPN").size(12.0).strong());
|
||||
ui.colored_label(TEXT2, egui::RichText::new("Обход для всех приложений").size(11.0));
|
||||
ui.with_layout(egui::Layout::right_to_left(egui::Align::Center), |ui| {
|
||||
if ui.add(egui::Button::new(
|
||||
egui::RichText::new("@bysonicvpn_bot").size(11.0).strong().color(ACCENT)
|
||||
).frame(false)).clicked() {
|
||||
let _ = open::that("https://t.me/bysonicvpn_bot");
|
||||
}
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
// --- Log panel ---
|
||||
// === Log (bottom) ===
|
||||
egui::TopBottomPanel::bottom("log")
|
||||
.min_height(130.0)
|
||||
.min_height(120.0)
|
||||
.show(ctx, |ui| {
|
||||
ui.label(egui::RichText::new("Лог").strong());
|
||||
ui.add_space(4.0);
|
||||
ui.colored_label(TEXT2, egui::RichText::new("LOG").size(11.0));
|
||||
ui.separator();
|
||||
egui::ScrollArea::vertical()
|
||||
.auto_shrink([false, false])
|
||||
.stick_to_bottom(true)
|
||||
.show(ui, |ui| {
|
||||
let logs = self.log.lock().unwrap();
|
||||
for e in logs.iter() {
|
||||
let color = if e.is_error {
|
||||
egui::Color32::from_rgb(255, 100, 100)
|
||||
} else {
|
||||
egui::Color32::from_rgb(170, 215, 170)
|
||||
};
|
||||
ui.colored_label(color, format!("[{}] {}", e.ts, e.text));
|
||||
for e in self.log.lock().unwrap().iter() {
|
||||
let c = if e.err { RED } else { TEXT2 };
|
||||
ui.colored_label(c, egui::RichText::new(
|
||||
format!("{} {}", e.ts, e.msg)
|
||||
).size(11.5).monospace());
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
// --- Main panel ---
|
||||
egui::CentralPanel::default().show(ctx, |ui| {
|
||||
ui.add_space(10.0);
|
||||
|
||||
// --- VPN ad (top) ---
|
||||
ui.vertical_centered(|ui| {
|
||||
egui::Frame::new()
|
||||
.fill(egui::Color32::from_rgb(25, 30, 42))
|
||||
.corner_radius(8.0)
|
||||
.inner_margin(egui::Margin::symmetric(14, 8))
|
||||
.show(ui, |ui| {
|
||||
ui.horizontal(|ui| {
|
||||
ui.colored_label(
|
||||
egui::Color32::from_rgb(100, 180, 255),
|
||||
egui::RichText::new("by sonic VPN").size(13.0).strong(),
|
||||
);
|
||||
ui.label(
|
||||
egui::RichText::new("Полный обход для всех приложений")
|
||||
.size(12.0)
|
||||
.color(egui::Color32::from_rgb(160, 165, 180)),
|
||||
);
|
||||
ui.with_layout(egui::Layout::right_to_left(egui::Align::Center), |ui| {
|
||||
if ui.add(
|
||||
egui::Button::new(
|
||||
egui::RichText::new("@bysonicvpn_bot")
|
||||
.size(12.0)
|
||||
.strong()
|
||||
.color(egui::Color32::from_rgb(100, 200, 255)),
|
||||
)
|
||||
.frame(false),
|
||||
).clicked() {
|
||||
let _ = open::that("https://t.me/bysonicvpn_bot");
|
||||
}
|
||||
});
|
||||
// === Stats bar ===
|
||||
egui::TopBottomPanel::bottom("stats").show(ctx, |ui| {
|
||||
egui::Frame::new()
|
||||
.fill(SURFACE)
|
||||
.inner_margin(egui::Margin::symmetric(16, 8))
|
||||
.show(ui, |ui| {
|
||||
ui.horizontal(|ui| {
|
||||
stat(ui, "Соединения", &active.to_string());
|
||||
ui.add_space(20.0);
|
||||
stat(ui, "WS-туннели", &ws.to_string());
|
||||
ui.add_space(20.0);
|
||||
stat(ui, "DC", &if dc > 0 { dc.to_string() } else { "—".into() });
|
||||
ui.add_space(20.0);
|
||||
stat(ui, "Всего", &total.to_string());
|
||||
ui.with_layout(egui::Layout::right_to_left(egui::Align::Center), |ui| {
|
||||
stat(ui, "Аптайм", &self.uptime_str());
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
ui.add_space(12.0);
|
||||
});
|
||||
});
|
||||
|
||||
// === Main ===
|
||||
egui::CentralPanel::default().show(ctx, |ui| {
|
||||
ui.vertical_centered(|ui| {
|
||||
if !running {
|
||||
ui.label(egui::RichText::new("Обход блокировки Telegram через WebSocket-прокси").size(15.0));
|
||||
ui.add_space(5.0);
|
||||
ui.label("Трафик идёт через web.telegram.org — провайдер видит обычный HTTPS");
|
||||
ui.add_space(15.0);
|
||||
ui.add_space(30.0);
|
||||
|
||||
// Title
|
||||
ui.colored_label(TEXT, egui::RichText::new("TGLock").size(32.0).strong());
|
||||
ui.add_space(4.0);
|
||||
ui.colored_label(TEXT2, egui::RichText::new("WebSocket-туннель для Telegram").size(13.0));
|
||||
|
||||
ui.add_space(24.0);
|
||||
|
||||
// Status indicator
|
||||
let (dot_color, status_text) = if on {
|
||||
(GREEN, "Подключено")
|
||||
} else {
|
||||
(egui::Color32::from_rgb(80, 80, 80), "Отключено")
|
||||
};
|
||||
|
||||
ui.horizontal(|ui| {
|
||||
let center = ui.available_width() / 2.0 - 50.0;
|
||||
ui.add_space(center);
|
||||
let (r, _) = ui.allocate_exact_size(egui::vec2(10.0, 10.0), egui::Sense::hover());
|
||||
ui.painter().circle_filled(r.center(), 5.0, dot_color);
|
||||
ui.colored_label(
|
||||
if on { GREEN } else { TEXT2 },
|
||||
egui::RichText::new(status_text).size(14.0).strong(),
|
||||
);
|
||||
});
|
||||
|
||||
ui.add_space(20.0);
|
||||
|
||||
// LAN toggle (only when stopped)
|
||||
if !on {
|
||||
ui.horizontal(|ui| {
|
||||
let center = ui.available_width() / 2.0 - 90.0;
|
||||
ui.add_space(center);
|
||||
ui.checkbox(&mut self.lan_mode, "");
|
||||
ui.colored_label(TEXT2, egui::RichText::new("Локальная сеть").size(12.0));
|
||||
ui.colored_label(
|
||||
egui::Color32::from_rgb(80, 85, 95),
|
||||
egui::RichText::new("(0.0.0.0 — для всех устройств)").size(10.5),
|
||||
);
|
||||
});
|
||||
ui.add_space(8.0);
|
||||
}
|
||||
|
||||
// Big button
|
||||
if !on {
|
||||
let btn = ui.add_sized(
|
||||
[340.0, 55.0],
|
||||
egui::Button::new(egui::RichText::new("Запустить обход").size(20.0).strong()),
|
||||
[260.0, 52.0],
|
||||
egui::Button::new(
|
||||
egui::RichText::new("ПОДКЛЮЧИТЬ").size(18.0).strong().color(BG)
|
||||
).fill(ACCENT).corner_radius(8.0),
|
||||
);
|
||||
if btn.clicked() {
|
||||
self.start_proxy();
|
||||
self.start();
|
||||
}
|
||||
} else {
|
||||
ui.colored_label(
|
||||
egui::Color32::from_rgb(80, 220, 120),
|
||||
egui::RichText::new("Обход работает").size(22.0).strong(),
|
||||
let btn = ui.add_sized(
|
||||
[260.0, 52.0],
|
||||
egui::Button::new(
|
||||
egui::RichText::new("ОТКЛЮЧИТЬ").size(18.0).strong().color(TEXT)
|
||||
).fill(egui::Color32::from_rgb(40, 45, 52)).corner_radius(8.0),
|
||||
);
|
||||
ui.add_space(5.0);
|
||||
ui.label(format!("SOCKS5 прокси на 127.0.0.1:{}", PROXY_PORT));
|
||||
ui.label(format!("WebSocket-туннелей: {} | Соединений: {}", ws, active));
|
||||
ui.add_space(12.0);
|
||||
|
||||
// Stop button
|
||||
let stop = ui.add_sized(
|
||||
[340.0, 42.0],
|
||||
egui::Button::new(egui::RichText::new("Остановить").size(17.0)),
|
||||
);
|
||||
if stop.clicked() {
|
||||
self.stop_proxy();
|
||||
if btn.clicked() {
|
||||
self.stop();
|
||||
}
|
||||
}
|
||||
|
||||
ui.add_space(24.0);
|
||||
|
||||
// Setup section
|
||||
egui::Frame::new()
|
||||
.fill(SURFACE)
|
||||
.corner_radius(8.0)
|
||||
.inner_margin(16.0)
|
||||
.show(ui, |ui| {
|
||||
ui.set_width(360.0);
|
||||
|
||||
ui.colored_label(TEXT, egui::RichText::new("Настройка Telegram").size(14.0).strong());
|
||||
ui.add_space(6.0);
|
||||
|
||||
let server_addr = if self.lan_mode && on {
|
||||
local_ip().unwrap_or_else(|| "127.0.0.1".into())
|
||||
} else {
|
||||
"127.0.0.1".into()
|
||||
};
|
||||
|
||||
if on {
|
||||
if ui.add(egui::Button::new(
|
||||
egui::RichText::new("Настроить автоматически").size(13.0).color(ACCENT)
|
||||
).frame(false)).clicked() {
|
||||
let _ = open::that(format!("tg://socks?server={}&port={}", server_addr, proxy::PORT));
|
||||
log(&self.log, "Открываю настройку Telegram...", false);
|
||||
}
|
||||
ui.add_space(4.0);
|
||||
}
|
||||
|
||||
ui.colored_label(TEXT2, egui::RichText::new("Настройки → Продвинутые → Тип соединения → SOCKS5").size(11.5));
|
||||
ui.add_space(4.0);
|
||||
|
||||
egui::Grid::new("cfg").num_columns(2).spacing([12.0, 3.0]).show(ui, |ui| {
|
||||
ui.colored_label(TEXT2, "Сервер");
|
||||
ui.monospace(&server_addr);
|
||||
ui.end_row();
|
||||
ui.colored_label(TEXT2, "Порт");
|
||||
ui.monospace(format!("{}", proxy::PORT));
|
||||
ui.end_row();
|
||||
});
|
||||
});
|
||||
|
||||
ui.add_space(16.0);
|
||||
|
||||
// How it works (compact)
|
||||
ui.colored_label(TEXT2, egui::RichText::new(
|
||||
"Трафик Telegram → SOCKS5 → WSS → web.telegram.org → DC"
|
||||
).size(11.0));
|
||||
ui.colored_label(TEXT2, egui::RichText::new(
|
||||
"Провайдер видит обычный HTTPS. Остальной трафик не затрагивается."
|
||||
).size(11.0));
|
||||
});
|
||||
|
||||
ui.add_space(20.0);
|
||||
ui.separator();
|
||||
ui.add_space(8.0);
|
||||
|
||||
// --- Telegram setup ---
|
||||
ui.heading("Настройка Telegram Desktop");
|
||||
ui.add_space(6.0);
|
||||
|
||||
if running {
|
||||
ui.horizontal(|ui| {
|
||||
if ui.button(" Настроить автоматически ").clicked() {
|
||||
self.open_tg_proxy_link();
|
||||
}
|
||||
ui.label("(откроет Telegram, нажмите \"Подключить\")");
|
||||
});
|
||||
ui.add_space(8.0);
|
||||
}
|
||||
|
||||
ui.label("Или вручную: Настройки → Продвинутые → Тип соединения → SOCKS5");
|
||||
ui.add_space(4.0);
|
||||
egui::Grid::new("manual_setup")
|
||||
.num_columns(2)
|
||||
.spacing([15.0, 4.0])
|
||||
.show(ui, |ui| {
|
||||
ui.label("Сервер:");
|
||||
ui.monospace("127.0.0.1");
|
||||
ui.end_row();
|
||||
ui.label("Порт:");
|
||||
ui.monospace(format!("{}", PROXY_PORT));
|
||||
ui.end_row();
|
||||
ui.label("Логин/Пароль:");
|
||||
ui.label("оставить пустыми");
|
||||
ui.end_row();
|
||||
});
|
||||
|
||||
ui.add_space(15.0);
|
||||
ui.separator();
|
||||
ui.add_space(5.0);
|
||||
|
||||
// --- How it works ---
|
||||
ui.heading("Как это работает");
|
||||
ui.add_space(4.0);
|
||||
ui.label("1. Локальный SOCKS5-прокси принимает соединения от Telegram");
|
||||
ui.label("2. Трафик к серверам Telegram заворачивается в WebSocket (WSS)");
|
||||
ui.label("3. Подключение идёт через web.telegram.org — обычный HTTPS");
|
||||
ui.label("4. Провайдер/DPI не видит MTProto, не может замедлить");
|
||||
ui.add_space(4.0);
|
||||
ui.colored_label(
|
||||
egui::Color32::from_rgb(170, 170, 170),
|
||||
"Не-Telegram трафик проходит напрямую без изменений",
|
||||
);
|
||||
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
fn stat(ui: &mut egui::Ui, label: &str, value: &str) {
|
||||
ui.vertical(|ui| {
|
||||
ui.colored_label(TEXT2, egui::RichText::new(label).size(10.0));
|
||||
ui.colored_label(TEXT, egui::RichText::new(value).size(13.0).strong().monospace());
|
||||
});
|
||||
}
|
||||
|
||||
fn local_ip() -> Option<String> {
|
||||
let socket = std::net::UdpSocket::bind("0.0.0.0:0").ok()?;
|
||||
socket.connect("8.8.8.8:80").ok()?;
|
||||
Some(socket.local_addr().ok()?.ip().to_string())
|
||||
}
|
||||
|
||||
176
src/network.rs
176
src/network.rs
@@ -1,176 +0,0 @@
|
||||
use std::net::{TcpStream, SocketAddr};
|
||||
use std::process::Command;
|
||||
use std::time::{Duration, Instant};
|
||||
|
||||
pub fn detect_adapter() -> Option<String> {
|
||||
let output = Command::new("powershell")
|
||||
.args([
|
||||
"-Command",
|
||||
"(Get-NetAdapter | Where-Object {$_.Status -eq 'Up'} | Select-Object -First 1).Name",
|
||||
])
|
||||
.output()
|
||||
.ok()?;
|
||||
|
||||
let name = String::from_utf8_lossy(&output.stdout).trim().to_string();
|
||||
if name.is_empty() {
|
||||
None
|
||||
} else {
|
||||
Some(name)
|
||||
}
|
||||
}
|
||||
|
||||
pub fn get_current_dns() -> Option<String> {
|
||||
let output = Command::new("powershell")
|
||||
.args([
|
||||
"-Command",
|
||||
"Get-DnsClientServerAddress -AddressFamily IPv4 | Where-Object {$_.ServerAddresses.Count -gt 0} | Select-Object -First 1 -ExpandProperty ServerAddresses | Out-String",
|
||||
])
|
||||
.output()
|
||||
.ok()?;
|
||||
|
||||
let result = String::from_utf8_lossy(&output.stdout).trim().to_string();
|
||||
if result.is_empty() {
|
||||
Some("Не определено".to_string())
|
||||
} else {
|
||||
Some(result.replace('\n', ", ").replace('\r', ""))
|
||||
}
|
||||
}
|
||||
|
||||
pub fn ping_host(ip: &str) -> (bool, Option<u64>) {
|
||||
let start = Instant::now();
|
||||
let output = Command::new("ping")
|
||||
.args(["-n", "1", "-w", "3000", ip])
|
||||
.output();
|
||||
|
||||
match output {
|
||||
Ok(out) => {
|
||||
let elapsed = start.elapsed().as_millis() as u64;
|
||||
let stdout = String::from_utf8_lossy(&out.stdout);
|
||||
let ok = out.status.success() && (stdout.contains("TTL=") || stdout.contains("ttl="));
|
||||
|
||||
if ok {
|
||||
// Try to extract actual time from ping output
|
||||
if let Some(time_str) = extract_ping_time(&stdout) {
|
||||
(true, Some(time_str))
|
||||
} else {
|
||||
(true, Some(elapsed))
|
||||
}
|
||||
} else {
|
||||
(false, None)
|
||||
}
|
||||
}
|
||||
Err(_) => (false, None),
|
||||
}
|
||||
}
|
||||
|
||||
fn extract_ping_time(output: &str) -> Option<u64> {
|
||||
// Match patterns like "time=46ms" or "time<1ms" or "время=46мс"
|
||||
for line in output.lines() {
|
||||
let lower = line.to_lowercase();
|
||||
if let Some(pos) = lower.find("time=").or_else(|| lower.find("time<")) {
|
||||
let after = &lower[pos + 5..];
|
||||
let num: String = after.chars().take_while(|c| c.is_ascii_digit()).collect();
|
||||
if let Ok(ms) = num.parse::<u64>() {
|
||||
return Some(ms);
|
||||
}
|
||||
}
|
||||
// Russian locale
|
||||
if let Some(pos) = lower.find("=").filter(|_| lower.contains("ms") || lower.contains("мс")) {
|
||||
let after = &lower[pos + 1..];
|
||||
let num: String = after.chars().take_while(|c| c.is_ascii_digit()).collect();
|
||||
if let Ok(ms) = num.parse::<u64>() {
|
||||
if ms < 10000 {
|
||||
return Some(ms);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
None
|
||||
}
|
||||
|
||||
pub fn tcp_check(ip: &str, port: u16) -> (bool, Option<u64>) {
|
||||
let addr: SocketAddr = format!("{}:{}", ip, port).parse().unwrap();
|
||||
let start = Instant::now();
|
||||
match TcpStream::connect_timeout(&addr, Duration::from_secs(5)) {
|
||||
Ok(_stream) => {
|
||||
let elapsed = start.elapsed().as_millis() as u64;
|
||||
(true, Some(elapsed))
|
||||
}
|
||||
Err(_) => (false, None),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn https_check(url: &str) -> (bool, Option<u64>) {
|
||||
let start = Instant::now();
|
||||
let client = reqwest::blocking::Client::builder()
|
||||
.timeout(Duration::from_secs(10))
|
||||
.danger_accept_invalid_certs(true)
|
||||
.build();
|
||||
|
||||
match client {
|
||||
Ok(c) => match c.get(url).send() {
|
||||
Ok(resp) => {
|
||||
let elapsed = start.elapsed().as_millis() as u64;
|
||||
(resp.status().is_success(), Some(elapsed))
|
||||
}
|
||||
Err(_) => (false, None),
|
||||
},
|
||||
Err(_) => (false, None),
|
||||
}
|
||||
}
|
||||
|
||||
/// Benchmarks Telegram connectivity: runs multiple TCP+HTTPS checks,
|
||||
/// returns (works: bool, score: u64) where lower score = faster connection.
|
||||
/// Score is average latency across all successful checks. u64::MAX if nothing works.
|
||||
pub fn benchmark_telegram() -> (bool, u64) {
|
||||
let tcp_targets = [
|
||||
("149.154.167.51", 443u16),
|
||||
("149.154.175.50", 443),
|
||||
("149.154.167.91", 443),
|
||||
("91.108.56.100", 443),
|
||||
];
|
||||
|
||||
let mut total_ms: u64 = 0;
|
||||
let mut ok_count: u64 = 0;
|
||||
let mut fail_count: u64 = 0;
|
||||
|
||||
// TCP checks (x2 rounds for stability)
|
||||
for _ in 0..2 {
|
||||
for (ip, port) in &tcp_targets {
|
||||
let (ok, latency) = tcp_check(ip, *port);
|
||||
if ok {
|
||||
total_ms += latency.unwrap_or(5000);
|
||||
ok_count += 1;
|
||||
} else {
|
||||
fail_count += 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// HTTPS check — the real indicator of usable speed
|
||||
let https_urls = [
|
||||
"https://web.telegram.org",
|
||||
"https://t.me",
|
||||
];
|
||||
for url in &https_urls {
|
||||
let (ok, latency) = https_check(url);
|
||||
if ok {
|
||||
// Weight HTTPS 3x heavier since it's closer to real usage
|
||||
let ms = latency.unwrap_or(10000);
|
||||
total_ms += ms * 3;
|
||||
ok_count += 3;
|
||||
} else {
|
||||
fail_count += 3;
|
||||
}
|
||||
}
|
||||
|
||||
if ok_count == 0 {
|
||||
return (false, u64::MAX);
|
||||
}
|
||||
|
||||
// Penalize failures: each fail adds 2000ms to the score
|
||||
let penalty = fail_count * 2000;
|
||||
let avg = (total_ms + penalty) / (ok_count + fail_count);
|
||||
|
||||
(true, avg)
|
||||
}
|
||||
233
src/proxy.rs
Normal file
233
src/proxy.rs
Normal file
@@ -0,0 +1,233 @@
|
||||
use std::net::Ipv4Addr;
|
||||
use std::sync::Arc;
|
||||
use std::sync::atomic::{AtomicBool, AtomicU32, AtomicU8, Ordering};
|
||||
use std::time::Duration;
|
||||
use tokio::net::{TcpListener, TcpStream};
|
||||
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
||||
use tokio_tungstenite::tungstenite;
|
||||
use tungstenite::client::IntoClientRequest;
|
||||
|
||||
pub const PORT: u16 = 1080;
|
||||
|
||||
pub struct Stats {
|
||||
pub running: AtomicBool,
|
||||
pub active: AtomicU32,
|
||||
pub total: AtomicU32,
|
||||
pub ws: AtomicU32,
|
||||
pub last_dc: AtomicU8,
|
||||
}
|
||||
|
||||
impl Stats {
|
||||
pub fn new() -> Arc<Self> {
|
||||
Arc::new(Self {
|
||||
running: AtomicBool::new(false),
|
||||
active: AtomicU32::new(0),
|
||||
total: AtomicU32::new(0),
|
||||
ws: AtomicU32::new(0),
|
||||
last_dc: AtomicU8::new(0),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
pub async fn run(stats: Arc<Stats>, lan: bool) -> Result<(), String> {
|
||||
let host = if lan { "0.0.0.0" } else { "127.0.0.1" };
|
||||
let addr = format!("{}:{}", host, PORT);
|
||||
let listener = TcpListener::bind(&addr)
|
||||
.await
|
||||
.map_err(|e| format!("Port {} busy: {}", PORT, e))?;
|
||||
|
||||
stats.running.store(true, Ordering::SeqCst);
|
||||
|
||||
loop {
|
||||
if !stats.running.load(Ordering::SeqCst) {
|
||||
break;
|
||||
}
|
||||
tokio::select! {
|
||||
Ok((stream, _)) = listener.accept() => {
|
||||
let s = stats.clone();
|
||||
s.active.fetch_add(1, Ordering::Relaxed);
|
||||
s.total.fetch_add(1, Ordering::Relaxed);
|
||||
tokio::spawn(async move {
|
||||
let _ = handle(stream, &s).await;
|
||||
s.active.fetch_sub(1, Ordering::Relaxed);
|
||||
});
|
||||
}
|
||||
_ = tokio::time::sleep(Duration::from_millis(150)) => {}
|
||||
}
|
||||
}
|
||||
|
||||
stats.running.store(false, Ordering::SeqCst);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
// -- SOCKS5 -----------------------------------------------------------------
|
||||
|
||||
async fn handle(
|
||||
mut s: TcpStream,
|
||||
stats: &Stats,
|
||||
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
||||
s.set_nodelay(true)?;
|
||||
|
||||
let mut buf = [0u8; 258];
|
||||
let n = s.read(&mut buf).await?;
|
||||
if n < 2 || buf[0] != 0x05 {
|
||||
return Err("not socks5".into());
|
||||
}
|
||||
s.write_all(&[0x05, 0x00]).await?;
|
||||
|
||||
let n = s.read(&mut buf).await?;
|
||||
if n < 7 || buf[0] != 0x05 || buf[1] != 0x01 {
|
||||
s.write_all(&[0x05, 0x07, 0x00, 0x01, 0, 0, 0, 0, 0, 0]).await?;
|
||||
return Err("bad connect".into());
|
||||
}
|
||||
|
||||
let (addr, port) = parse_addr(&buf[3..n])?;
|
||||
let tg = addr.parse::<Ipv4Addr>().ok().and_then(dc_from_ip).is_some();
|
||||
|
||||
// success reply
|
||||
s.write_all(&[0x05, 0x00, 0x00, 0x01, 127, 0, 0, 1, 0x04, 0x38]).await?;
|
||||
|
||||
if tg {
|
||||
// Read 64-byte obfuscated2 init → extract real DC
|
||||
let mut init = [0u8; 64];
|
||||
s.read_exact(&mut init).await?;
|
||||
|
||||
let dc = dc_from_init(&init).unwrap_or_else(|| {
|
||||
addr.parse::<Ipv4Addr>().ok().and_then(dc_from_ip).unwrap_or(2)
|
||||
});
|
||||
|
||||
stats.last_dc.store(dc, Ordering::Relaxed);
|
||||
stats.ws.fetch_add(1, Ordering::Relaxed);
|
||||
|
||||
let r = ws_tunnel(s, dc, &init).await;
|
||||
|
||||
stats.ws.fetch_sub(1, Ordering::Relaxed);
|
||||
r?;
|
||||
} else {
|
||||
let remote = TcpStream::connect(format!("{}:{}", addr, port)).await?;
|
||||
let _ = remote.set_nodelay(true);
|
||||
tcp_relay(s, remote).await;
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn parse_addr(d: &[u8]) -> Result<(String, u16), Box<dyn std::error::Error + Send + Sync>> {
|
||||
match d[0] {
|
||||
0x01 if d.len() >= 7 => {
|
||||
Ok((format!("{}.{}.{}.{}", d[1], d[2], d[3], d[4]),
|
||||
u16::from_be_bytes([d[5], d[6]])))
|
||||
}
|
||||
0x03 => {
|
||||
let l = d[1] as usize;
|
||||
if d.len() < 2 + l + 2 { return Err("short".into()); }
|
||||
Ok((std::str::from_utf8(&d[2..2 + l])?.into(),
|
||||
u16::from_be_bytes([d[2 + l], d[3 + l]])))
|
||||
}
|
||||
0x04 if d.len() >= 19 => {
|
||||
let mut seg = [0u16; 8];
|
||||
for i in 0..8 { seg[i] = u16::from_be_bytes([d[1 + i * 2], d[2 + i * 2]]); }
|
||||
let ip = std::net::Ipv6Addr::new(seg[0],seg[1],seg[2],seg[3],seg[4],seg[5],seg[6],seg[7]);
|
||||
Ok((ip.to_string(), u16::from_be_bytes([d[17], d[18]])))
|
||||
}
|
||||
_ => Err("bad addr".into()),
|
||||
}
|
||||
}
|
||||
|
||||
// -- DC detection -----------------------------------------------------------
|
||||
|
||||
fn dc_from_init(init: &[u8; 64]) -> Option<u8> {
|
||||
use aes::Aes256;
|
||||
use cipher::{KeyIvInit, StreamCipher};
|
||||
type Ctr = ctr::Ctr128BE<Aes256>;
|
||||
|
||||
let mut dec = *init;
|
||||
let mut c = Ctr::new(init[8..40].into(), init[40..56].into());
|
||||
c.apply_keystream(&mut dec);
|
||||
|
||||
let id = i32::from_le_bytes([dec[60], dec[61], dec[62], dec[63]]);
|
||||
let dc = id.unsigned_abs() as u8;
|
||||
(1..=5).contains(&dc).then_some(dc)
|
||||
}
|
||||
|
||||
fn dc_from_ip(ip: Ipv4Addr) -> Option<u8> {
|
||||
let o = ip.octets();
|
||||
match (o[0], o[1]) {
|
||||
(149, 154) => Some(match o[2] { 160..=163 => 1, 164..=167 => 2, 168..=171 => 3, 172..=175 => 1, _ => 2 }),
|
||||
(91, 108) => Some(match o[2] { 56..=59 => 5, 8..=11 => 3, 12..=15 => 4, _ => 2 }),
|
||||
(91, 105) | (185, 76) => Some(2),
|
||||
_ => None,
|
||||
}
|
||||
}
|
||||
|
||||
fn ws_url(dc: u8) -> String {
|
||||
format!("wss://kws{}.web.telegram.org/apiws", dc)
|
||||
}
|
||||
|
||||
// -- WebSocket tunnel -------------------------------------------------------
|
||||
|
||||
async fn ws_tunnel(
|
||||
tcp: TcpStream,
|
||||
dc: u8,
|
||||
init: &[u8; 64],
|
||||
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
||||
use futures_util::{SinkExt, StreamExt};
|
||||
|
||||
let mut req = ws_url(dc).as_str().into_client_request()?;
|
||||
req.headers_mut().insert("Sec-WebSocket-Protocol", "binary".parse()?);
|
||||
|
||||
let tls = native_tls::TlsConnector::new().map_err(|e| format!("tls: {}", e))?;
|
||||
let connector = tokio_tungstenite::Connector::NativeTls(tls);
|
||||
|
||||
let (mut ws, _) = tokio::time::timeout(
|
||||
Duration::from_secs(10),
|
||||
tokio_tungstenite::connect_async_tls_with_config(req, None, false, Some(connector)),
|
||||
)
|
||||
.await
|
||||
.map_err(|_| "WS connect timeout")?
|
||||
.map_err(|e| format!("WS: {}", e))?;
|
||||
|
||||
let (mut tcp_r, mut tcp_w) = tokio::io::split(tcp);
|
||||
|
||||
// Send buffered init as first frame
|
||||
ws.send(tungstenite::Message::Binary(init.to_vec())).await?;
|
||||
|
||||
let mut buf = vec![0u8; 65536];
|
||||
|
||||
loop {
|
||||
tokio::select! {
|
||||
biased;
|
||||
|
||||
msg = ws.next() => match msg {
|
||||
Some(Ok(tungstenite::Message::Binary(data))) => {
|
||||
tcp_w.write_all(data.as_ref()).await?;
|
||||
tcp_w.flush().await?;
|
||||
}
|
||||
Some(Ok(tungstenite::Message::Ping(p))) => {
|
||||
let _ = ws.send(tungstenite::Message::Pong(p)).await;
|
||||
}
|
||||
Some(Ok(tungstenite::Message::Close(_))) | None => break,
|
||||
Some(Err(_)) => break,
|
||||
_ => {}
|
||||
},
|
||||
|
||||
n = tcp_r.read(&mut buf) => match n {
|
||||
Ok(0) | Err(_) => break,
|
||||
Ok(n) => {
|
||||
ws.send(tungstenite::Message::Binary(buf[..n].to_vec())).await?;
|
||||
}
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
let _ = ws.close(None).await;
|
||||
Ok(())
|
||||
}
|
||||
|
||||
async fn tcp_relay(a: TcpStream, b: TcpStream) {
|
||||
let (mut ar, mut aw) = tokio::io::split(a);
|
||||
let (mut br, mut bw) = tokio::io::split(b);
|
||||
tokio::select! {
|
||||
_ = tokio::io::copy(&mut ar, &mut bw) => {}
|
||||
_ = tokio::io::copy(&mut br, &mut aw) => {}
|
||||
}
|
||||
}
|
||||
306
src/ws_proxy.rs
306
src/ws_proxy.rs
@@ -1,306 +0,0 @@
|
||||
use std::net::Ipv4Addr;
|
||||
use std::sync::Arc;
|
||||
use std::sync::atomic::{AtomicBool, AtomicU32, Ordering};
|
||||
use tokio::net::{TcpListener, TcpStream};
|
||||
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
||||
use tokio_tungstenite::tungstenite;
|
||||
use tungstenite::client::IntoClientRequest;
|
||||
|
||||
pub struct ProxyStats {
|
||||
pub running: AtomicBool,
|
||||
pub active_conn: AtomicU32,
|
||||
pub total_conn: AtomicU32,
|
||||
pub ws_active: AtomicU32,
|
||||
}
|
||||
|
||||
impl ProxyStats {
|
||||
pub fn new() -> Arc<Self> {
|
||||
Arc::new(Self {
|
||||
running: AtomicBool::new(false),
|
||||
active_conn: AtomicU32::new(0),
|
||||
total_conn: AtomicU32::new(0),
|
||||
ws_active: AtomicU32::new(0),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
pub async fn run_proxy(port: u16, stats: Arc<ProxyStats>) -> Result<(), String> {
|
||||
let addr = format!("127.0.0.1:{}", port);
|
||||
let listener = TcpListener::bind(&addr)
|
||||
.await
|
||||
.map_err(|e| format!("Не удалось занять порт {}: {}", port, e))?;
|
||||
|
||||
stats.running.store(true, Ordering::SeqCst);
|
||||
|
||||
loop {
|
||||
if !stats.running.load(Ordering::SeqCst) {
|
||||
break;
|
||||
}
|
||||
tokio::select! {
|
||||
result = listener.accept() => {
|
||||
if let Ok((stream, _)) = result {
|
||||
let st = stats.clone();
|
||||
st.active_conn.fetch_add(1, Ordering::Relaxed);
|
||||
st.total_conn.fetch_add(1, Ordering::Relaxed);
|
||||
tokio::spawn(async move {
|
||||
let _ = handle_socks5(stream, &st).await;
|
||||
st.active_conn.fetch_sub(1, Ordering::Relaxed);
|
||||
});
|
||||
}
|
||||
}
|
||||
_ = tokio::time::sleep(std::time::Duration::from_millis(200)) => {}
|
||||
}
|
||||
}
|
||||
|
||||
stats.running.store(false, Ordering::SeqCst);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// DC extraction from obfuscated2 init packet (same method as tg-ws-proxy)
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
fn extract_dc_from_init(init: &[u8; 64]) -> Option<u8> {
|
||||
use aes::Aes256;
|
||||
use cipher::{KeyIvInit, StreamCipher};
|
||||
type Aes256Ctr = ctr::Ctr128BE<Aes256>;
|
||||
|
||||
let key = &init[8..40];
|
||||
let iv = &init[40..56];
|
||||
|
||||
let mut dec = [0u8; 64];
|
||||
dec.copy_from_slice(init);
|
||||
|
||||
let mut cipher = Aes256Ctr::new(key.into(), iv.into());
|
||||
cipher.apply_keystream(&mut dec);
|
||||
|
||||
let dc_id = i32::from_le_bytes([dec[60], dec[61], dec[62], dec[63]]);
|
||||
let dc = dc_id.unsigned_abs() as u8;
|
||||
if (1..=5).contains(&dc) {
|
||||
Some(dc)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
fn dc_from_ip(ip: Ipv4Addr) -> Option<u8> {
|
||||
let o = ip.octets();
|
||||
match (o[0], o[1]) {
|
||||
(149, 154) => Some(match o[2] {
|
||||
160..=163 => 1,
|
||||
164..=167 => 2,
|
||||
168..=171 => 3,
|
||||
172..=175 => 1,
|
||||
_ => 2,
|
||||
}),
|
||||
(91, 108) => Some(match o[2] {
|
||||
56..=59 => 5,
|
||||
8..=11 => 3,
|
||||
12..=15 => 4,
|
||||
_ => 2,
|
||||
}),
|
||||
(91, 105) | (185, 76) => Some(2),
|
||||
_ => None,
|
||||
}
|
||||
}
|
||||
|
||||
fn is_telegram_ip(addr: &str) -> bool {
|
||||
addr.parse::<Ipv4Addr>()
|
||||
.ok()
|
||||
.and_then(dc_from_ip)
|
||||
.is_some()
|
||||
}
|
||||
|
||||
/// Endpoint format used by the proven tg-ws-proxy project
|
||||
fn ws_url(dc: u8) -> String {
|
||||
format!("wss://kws{}.web.telegram.org/apiws", dc)
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// SOCKS5 handler
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
async fn handle_socks5(
|
||||
mut stream: TcpStream,
|
||||
stats: &ProxyStats,
|
||||
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
||||
stream.set_nodelay(true)?;
|
||||
|
||||
// --- auth negotiation ---
|
||||
let mut buf = [0u8; 258];
|
||||
let n = stream.read(&mut buf).await?;
|
||||
if n < 2 || buf[0] != 0x05 {
|
||||
return Err("Not SOCKS5".into());
|
||||
}
|
||||
stream.write_all(&[0x05, 0x00]).await?;
|
||||
|
||||
// --- CONNECT request ---
|
||||
let n = stream.read(&mut buf).await?;
|
||||
if n < 7 || buf[0] != 0x05 || buf[1] != 0x01 {
|
||||
stream.write_all(&[0x05, 0x07, 0x00, 0x01, 0, 0, 0, 0, 0, 0]).await?;
|
||||
return Err("Bad CONNECT".into());
|
||||
}
|
||||
|
||||
let (dest_addr, dest_port) = parse_dest(&buf[3..n])?;
|
||||
let is_tg = is_telegram_ip(&dest_addr);
|
||||
|
||||
// SOCKS5 success (we handle the connection ourselves)
|
||||
stream
|
||||
.write_all(&[0x05, 0x00, 0x00, 0x01, 127, 0, 0, 1, 0x04, 0x38])
|
||||
.await?;
|
||||
|
||||
if is_tg {
|
||||
// Read the first 64 bytes — obfuscated2 init packet
|
||||
let mut init = [0u8; 64];
|
||||
stream.read_exact(&mut init).await?;
|
||||
|
||||
// Extract DC from init packet (primary), fall back to IP-based
|
||||
let dc = extract_dc_from_init(&init).unwrap_or_else(|| {
|
||||
dest_addr
|
||||
.parse::<Ipv4Addr>()
|
||||
.ok()
|
||||
.and_then(dc_from_ip)
|
||||
.unwrap_or(2)
|
||||
});
|
||||
|
||||
stats.ws_active.fetch_add(1, Ordering::Relaxed);
|
||||
|
||||
// Try WebSocket tunnel; fall back to direct TCP on failure
|
||||
let ws_result = relay_via_ws(stream, dc, &init).await;
|
||||
|
||||
stats.ws_active.fetch_sub(1, Ordering::Relaxed);
|
||||
|
||||
if let Err(e) = ws_result {
|
||||
return Err(format!("DC{} tunnel: {}", dc, e).into());
|
||||
}
|
||||
} else {
|
||||
// Non-Telegram — direct TCP passthrough
|
||||
let target = format!("{}:{}", dest_addr, dest_port);
|
||||
match TcpStream::connect(&target).await {
|
||||
Ok(remote) => {
|
||||
let _ = remote.set_nodelay(true);
|
||||
relay_tcp(stream, remote).await;
|
||||
}
|
||||
Err(e) => {
|
||||
return Err(format!("TCP connect {}: {}", target, e).into());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn parse_dest(data: &[u8]) -> Result<(String, u16), Box<dyn std::error::Error + Send + Sync>> {
|
||||
match data[0] {
|
||||
0x01 => {
|
||||
if data.len() < 7 { return Err("short".into()); }
|
||||
let ip = format!("{}.{}.{}.{}", data[1], data[2], data[3], data[4]);
|
||||
let port = u16::from_be_bytes([data[5], data[6]]);
|
||||
Ok((ip, port))
|
||||
}
|
||||
0x03 => {
|
||||
let len = data[1] as usize;
|
||||
if data.len() < 2 + len + 2 { return Err("short".into()); }
|
||||
let domain = std::str::from_utf8(&data[2..2 + len])?.to_string();
|
||||
let port = u16::from_be_bytes([data[2 + len], data[3 + len]]);
|
||||
Ok((domain, port))
|
||||
}
|
||||
0x04 => {
|
||||
if data.len() < 19 { return Err("short".into()); }
|
||||
let port = u16::from_be_bytes([data[17], data[18]]);
|
||||
let mut segs = [0u16; 8];
|
||||
for i in 0..8 {
|
||||
segs[i] = u16::from_be_bytes([data[1 + i * 2], data[2 + i * 2]]);
|
||||
}
|
||||
let ip = std::net::Ipv6Addr::new(
|
||||
segs[0], segs[1], segs[2], segs[3], segs[4], segs[5], segs[6], segs[7],
|
||||
);
|
||||
Ok((ip.to_string(), port))
|
||||
}
|
||||
_ => Err("unknown addr type".into()),
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// WebSocket tunnel — reads init first, then relays
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
async fn relay_via_ws(
|
||||
tcp_stream: TcpStream,
|
||||
dc: u8,
|
||||
init: &[u8; 64],
|
||||
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
|
||||
use futures_util::{SinkExt, StreamExt};
|
||||
|
||||
let url = ws_url(dc);
|
||||
let mut request = url.as_str().into_client_request()?;
|
||||
|
||||
request
|
||||
.headers_mut()
|
||||
.insert("Sec-WebSocket-Protocol", "binary".parse()?);
|
||||
|
||||
let connector = tokio_tungstenite::Connector::NativeTls(
|
||||
native_tls::TlsConnector::new().map_err(|e| format!("TLS: {}", e))?,
|
||||
);
|
||||
|
||||
let (mut ws, _resp) = tokio_tungstenite::connect_async_tls_with_config(
|
||||
request, None, false, Some(connector),
|
||||
)
|
||||
.await?;
|
||||
|
||||
let (mut tcp_rx, mut tcp_tx) = tokio::io::split(tcp_stream);
|
||||
|
||||
// Send the buffered 64-byte init as the first WebSocket message
|
||||
ws.send(tungstenite::Message::Binary(init.to_vec())).await?;
|
||||
|
||||
// Single loop: handles TCP→WS, WS→TCP, and Ping/Pong in one place.
|
||||
// This ensures Pong replies are sent immediately so the server
|
||||
// doesn't kill the connection after a timeout.
|
||||
let mut buf = vec![0u8; 32768];
|
||||
|
||||
loop {
|
||||
tokio::select! {
|
||||
biased;
|
||||
|
||||
ws_msg = ws.next() => {
|
||||
match ws_msg {
|
||||
Some(Ok(tungstenite::Message::Binary(data))) => {
|
||||
if tcp_tx.write_all(data.as_ref()).await.is_err() {
|
||||
break;
|
||||
}
|
||||
}
|
||||
Some(Ok(tungstenite::Message::Ping(payload))) => {
|
||||
let _ = ws.send(tungstenite::Message::Pong(payload)).await;
|
||||
}
|
||||
Some(Ok(tungstenite::Message::Close(_))) | None => break,
|
||||
Some(Err(_)) => break,
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
|
||||
n = tcp_rx.read(&mut buf) => {
|
||||
match n {
|
||||
Ok(0) | Err(_) => break,
|
||||
Ok(n) => {
|
||||
let msg = tungstenite::Message::Binary(buf[..n].to_vec());
|
||||
if ws.send(msg).await.is_err() {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
let _ = ws.close(None).await;
|
||||
Ok(())
|
||||
}
|
||||
|
||||
async fn relay_tcp(client: TcpStream, remote: TcpStream) {
|
||||
let (mut cr, mut cw) = tokio::io::split(client);
|
||||
let (mut rr, mut rw) = tokio::io::split(remote);
|
||||
tokio::select! {
|
||||
_ = tokio::io::copy(&mut cr, &mut rw) => {}
|
||||
_ = tokio::io::copy(&mut rr, &mut cw) => {}
|
||||
}
|
||||
}
|
||||
@@ -1,21 +0,0 @@
|
||||
91.108.56.0/22
|
||||
91.108.4.0/22
|
||||
91.108.8.0/22
|
||||
91.108.16.0/22
|
||||
91.108.12.0/22
|
||||
91.108.20.0/22
|
||||
149.154.160.0/20
|
||||
185.76.151.0/24
|
||||
91.105.192.0/23
|
||||
core.telegram.org
|
||||
web.telegram.org
|
||||
desktop.telegram.org
|
||||
macos.telegram.org
|
||||
updates.telegram.org
|
||||
venus.web.telegram.org
|
||||
pluto.web.telegram.org
|
||||
flora.web.telegram.org
|
||||
vesta.web.telegram.org
|
||||
telegram.org
|
||||
t.me
|
||||
telegram.me
|
||||
852
tg_unblock.bat
852
tg_unblock.bat
@@ -1,852 +0,0 @@
|
||||
@echo off
|
||||
chcp 65001 >nul 2>&1
|
||||
setlocal enabledelayedexpansion
|
||||
|
||||
:: ============================================================
|
||||
:: TG Unblock — обход блокировки/замедления Telegram
|
||||
:: Требуется запуск от имени администратора
|
||||
:: ============================================================
|
||||
|
||||
:: --- Проверка прав администратора ---
|
||||
net session >nul 2>&1
|
||||
if %errorlevel% neq 0 (
|
||||
echo [!] Скрипт требует прав администратора.
|
||||
echo Перезапускаю с повышенными правами...
|
||||
powershell -Command "Start-Process -Verb RunAs -FilePath '%~f0'"
|
||||
exit /b
|
||||
)
|
||||
|
||||
set "SCRIPT_DIR=%~dp0"
|
||||
set "TOOLS_DIR=%SCRIPT_DIR%tools"
|
||||
set "GDPI_DIR=%TOOLS_DIR%\goodbyedpi-0.2.3rc3-2"
|
||||
set "GDPI_EXE=%GDPI_DIR%\x86_64\goodbyedpi.exe"
|
||||
set "BLACKLIST=%SCRIPT_DIR%tg_blacklist.txt"
|
||||
set "GDPI_URL=https://github.com/ValdikSS/GoodbyeDPI/releases/download/0.2.3rc3/goodbyedpi-0.2.3rc3-2.zip"
|
||||
set "GDPI_ZIP=%TOOLS_DIR%\goodbyedpi.zip"
|
||||
|
||||
title TG Unblock — Обход блокировки Telegram
|
||||
|
||||
:MENU
|
||||
cls
|
||||
echo ============================================================
|
||||
echo TG Unblock — Обход блокировки Telegram
|
||||
echo ============================================================
|
||||
echo.
|
||||
echo [*] ЗАПУСТИТЬ ОБХОД (авто) — нажмите 7
|
||||
echo.
|
||||
echo [1] Сменить DNS (Cloudflare / Google)
|
||||
echo [2] Запустить GoodbyeDPI (обход DPI)
|
||||
echo [3] Комбинированный режим (DNS + GoodbyeDPI)
|
||||
echo [4] Тест соединения с Telegram
|
||||
echo [5] Сбросить настройки (вернуть DNS, остановить GoodbyeDPI)
|
||||
echo [6] Показать текущие настройки сети
|
||||
echo [7] === АВТО-ОБХОД (одна кнопка) ===
|
||||
echo [0] Выход
|
||||
echo.
|
||||
echo ============================================================
|
||||
set /p "choice=Выберите действие [0-7]: "
|
||||
|
||||
if "%choice%"=="1" goto DNS_MENU
|
||||
if "%choice%"=="2" goto DPI_START
|
||||
if "%choice%"=="3" goto COMBINED
|
||||
if "%choice%"=="4" goto TEST
|
||||
if "%choice%"=="5" goto RESET
|
||||
if "%choice%"=="6" goto SHOW_NET
|
||||
if "%choice%"=="7" goto AUTO_BYPASS
|
||||
if "%choice%"=="0" goto EXIT
|
||||
echo [!] Неверный выбор.
|
||||
timeout /t 2 >nul
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: 1. СМЕНА DNS
|
||||
:: ============================================================
|
||||
:DNS_MENU
|
||||
cls
|
||||
echo ============================================================
|
||||
echo Смена DNS
|
||||
echo ============================================================
|
||||
echo.
|
||||
echo [1] Cloudflare DNS (1.1.1.1 / 1.0.0.1)
|
||||
echo [2] Google DNS (8.8.8.8 / 8.8.4.4)
|
||||
echo [3] Quad9 DNS (9.9.9.9 / 149.112.112.112)
|
||||
echo [4] Cloudflare DoH (1.1.1.1 + DNS-over-HTTPS)
|
||||
echo [0] Назад
|
||||
echo.
|
||||
set /p "dns_choice=Выберите DNS провайдера [0-4]: "
|
||||
|
||||
if "%dns_choice%"=="1" (
|
||||
set "DNS1=1.1.1.1"
|
||||
set "DNS2=1.0.0.1"
|
||||
set "DNS_NAME=Cloudflare"
|
||||
)
|
||||
if "%dns_choice%"=="2" (
|
||||
set "DNS1=8.8.8.8"
|
||||
set "DNS2=8.8.4.4"
|
||||
set "DNS_NAME=Google"
|
||||
)
|
||||
if "%dns_choice%"=="3" (
|
||||
set "DNS1=9.9.9.9"
|
||||
set "DNS2=149.112.112.112"
|
||||
set "DNS_NAME=Quad9"
|
||||
)
|
||||
if "%dns_choice%"=="4" (
|
||||
set "DNS1=1.1.1.1"
|
||||
set "DNS2=1.0.0.1"
|
||||
set "DNS_NAME=Cloudflare DoH"
|
||||
goto SET_DNS_DOH
|
||||
)
|
||||
if "%dns_choice%"=="0" goto MENU
|
||||
|
||||
if not defined DNS1 (
|
||||
echo [!] Неверный выбор.
|
||||
timeout /t 2 >nul
|
||||
goto DNS_MENU
|
||||
)
|
||||
|
||||
:SET_DNS
|
||||
echo.
|
||||
echo [*] Определяю активный сетевой адаптер...
|
||||
|
||||
for /f "tokens=1,2,3,4,*" %%a in ('netsh interface ipv4 show interfaces ^| findstr /i "connected"') do (
|
||||
set "ADAPTER_NAME=%%e"
|
||||
)
|
||||
|
||||
if not defined ADAPTER_NAME (
|
||||
for /f "tokens=*" %%a in ('powershell -Command "(Get-NetAdapter | Where-Object {$_.Status -eq 'Up'} | Select-Object -First 1).Name"') do (
|
||||
set "ADAPTER_NAME=%%a"
|
||||
)
|
||||
)
|
||||
|
||||
if not defined ADAPTER_NAME (
|
||||
echo [!] Не удалось определить сетевой адаптер.
|
||||
echo Введите имя адаптера вручную (например: Ethernet, Wi-Fi):
|
||||
set /p "ADAPTER_NAME="
|
||||
)
|
||||
|
||||
echo [*] Адаптер: !ADAPTER_NAME!
|
||||
echo [*] Устанавливаю DNS: %DNS_NAME% (%DNS1%, %DNS2%)...
|
||||
|
||||
netsh interface ipv4 set dnsservers "!ADAPTER_NAME!" static %DNS1% primary validate=no >nul 2>&1
|
||||
netsh interface ipv4 add dnsservers "!ADAPTER_NAME!" %DNS2% index=2 validate=no >nul 2>&1
|
||||
|
||||
echo [*] Очищаю DNS-кеш...
|
||||
ipconfig /flushdns >nul 2>&1
|
||||
|
||||
echo.
|
||||
echo [OK] DNS успешно изменён на %DNS_NAME% (%DNS1%, %DNS2%)
|
||||
echo.
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:SET_DNS_DOH
|
||||
echo.
|
||||
echo [*] Определяю активный сетевой адаптер...
|
||||
|
||||
for /f "tokens=*" %%a in ('powershell -Command "(Get-NetAdapter | Where-Object {$_.Status -eq 'Up'} | Select-Object -First 1).Name"') do (
|
||||
set "ADAPTER_NAME=%%a"
|
||||
)
|
||||
|
||||
if not defined ADAPTER_NAME (
|
||||
echo [!] Не удалось определить адаптер. Введите имя вручную:
|
||||
set /p "ADAPTER_NAME="
|
||||
)
|
||||
|
||||
echo [*] Адаптер: !ADAPTER_NAME!
|
||||
echo [*] Устанавливаю DNS: Cloudflare (1.1.1.1, 1.0.0.1)...
|
||||
|
||||
netsh interface ipv4 set dnsservers "!ADAPTER_NAME!" static 1.1.1.1 primary validate=no >nul 2>&1
|
||||
netsh interface ipv4 add dnsservers "!ADAPTER_NAME!" 1.0.0.1 index=2 validate=no >nul 2>&1
|
||||
ipconfig /flushdns >nul 2>&1
|
||||
|
||||
echo [*] Включаю DNS-over-HTTPS в системе (Windows 11+)...
|
||||
powershell -Command "try { Set-DnsClientDohServerAddress -ServerAddress '1.1.1.1' -DohTemplate 'https://cloudflare-dns.com/dns-query' -AllowFallbackToUdp $true -AutoUpgrade $true -ErrorAction Stop; Set-DnsClientDohServerAddress -ServerAddress '1.0.0.1' -DohTemplate 'https://cloudflare-dns.com/dns-query' -AllowFallbackToUdp $true -AutoUpgrade $true -ErrorAction Stop; Write-Host '[OK] DoH активирован' } catch { Write-Host '[!] DoH недоступен на этой версии Windows, DNS установлен без DoH' }"
|
||||
|
||||
echo.
|
||||
echo [OK] DNS установлен на Cloudflare с DoH.
|
||||
echo.
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: 2. ЗАПУСК GoodbyeDPI
|
||||
:: ============================================================
|
||||
:DPI_START
|
||||
cls
|
||||
echo ============================================================
|
||||
echo GoodbyeDPI — обход DPI
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
:: Проверяем наличие GoodbyeDPI
|
||||
if not exist "%GDPI_EXE%" (
|
||||
echo [!] GoodbyeDPI не найден.
|
||||
goto DOWNLOAD_GDPI
|
||||
)
|
||||
|
||||
:DPI_MODE_MENU
|
||||
echo.
|
||||
echo Выберите режим GoodbyeDPI:
|
||||
echo.
|
||||
echo [1] Режим -9 (максимальная защита, рекомендуется)
|
||||
echo [2] Режим -5 (средний)
|
||||
echo [3] Режим -1 (базовый, совместимый)
|
||||
echo [4] Кастомный (-e 2 -f 2 -q -r -m)
|
||||
echo [5] Авто-перебор (попробует все режимы)
|
||||
echo [0] Назад
|
||||
echo.
|
||||
set /p "dpi_mode=Выберите режим [0-5]: "
|
||||
|
||||
if "%dpi_mode%"=="0" goto MENU
|
||||
|
||||
:: Убиваем старый процесс если запущен
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
|
||||
if "%dpi_mode%"=="1" (
|
||||
echo [*] Запускаю GoodbyeDPI в режиме -9 (максимальная защита)...
|
||||
start "" /B "%GDPI_EXE%" -9 --blacklist "%BLACKLIST%"
|
||||
goto DPI_STARTED
|
||||
)
|
||||
if "%dpi_mode%"=="2" (
|
||||
echo [*] Запускаю GoodbyeDPI в режиме -5 (средний)...
|
||||
start "" /B "%GDPI_EXE%" -5 --blacklist "%BLACKLIST%"
|
||||
goto DPI_STARTED
|
||||
)
|
||||
if "%dpi_mode%"=="3" (
|
||||
echo [*] Запускаю GoodbyeDPI в режиме -1 (базовый)...
|
||||
start "" /B "%GDPI_EXE%" -1 --blacklist "%BLACKLIST%"
|
||||
goto DPI_STARTED
|
||||
)
|
||||
if "%dpi_mode%"=="4" (
|
||||
echo [*] Запускаю GoodbyeDPI в кастомном режиме...
|
||||
start "" /B "%GDPI_EXE%" -e 2 -f 2 -q -r -m --blacklist "%BLACKLIST%"
|
||||
goto DPI_STARTED
|
||||
)
|
||||
if "%dpi_mode%"=="5" goto DPI_AUTO
|
||||
|
||||
echo [!] Неверный выбор.
|
||||
timeout /t 2 >nul
|
||||
goto DPI_MODE_MENU
|
||||
|
||||
:DPI_STARTED
|
||||
timeout /t 2 >nul
|
||||
tasklist /fi "imagename eq goodbyedpi.exe" 2>nul | findstr /i "goodbyedpi" >nul
|
||||
if %errorlevel%==0 (
|
||||
echo.
|
||||
echo [OK] GoodbyeDPI запущен успешно!
|
||||
echo Работает в фоне. Для остановки используйте пункт 5 (Сброс).
|
||||
) else (
|
||||
echo.
|
||||
echo [!] GoodbyeDPI не удалось запустить.
|
||||
echo Проверьте что антивирус не блокирует WinDivert.
|
||||
)
|
||||
echo.
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: АВТО-ПЕРЕБОР РЕЖИМОВ
|
||||
:: ============================================================
|
||||
:DPI_AUTO
|
||||
echo.
|
||||
echo [*] Авто-перебор режимов GoodbyeDPI...
|
||||
echo [*] Буду пробовать каждый режим и тестировать соединение.
|
||||
echo.
|
||||
|
||||
set "MODES=-9;-5;-1;-e 2 -f 2 -q -r -m;-e 1 -f 1 -q -r -m -s;-p -f 2 -e 2 -q"
|
||||
set "MODE_NAMES=Режим -9 (макс);Режим -5 (средний);Режим -1 (базовый);Кастом1 (-e2 -f2 -q -r -m);Кастом2 (-e1 -f1 -q -r -m -s);Кастом3 (-p -f2 -e2 -q)"
|
||||
set "mode_idx=0"
|
||||
set "best_mode="
|
||||
|
||||
for %%m in ("-9" "-5" "-1") do (
|
||||
set /a mode_idx+=1
|
||||
echo --- Тест !mode_idx!: GoodbyeDPI %%~m ---
|
||||
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
|
||||
start "" /B "%GDPI_EXE%" %%~m --blacklist "%BLACKLIST%"
|
||||
timeout /t 3 >nul
|
||||
|
||||
tasklist /fi "imagename eq goodbyedpi.exe" 2>nul | findstr /i "goodbyedpi" >nul
|
||||
if !errorlevel! neq 0 (
|
||||
echo [FAIL] Не запустился
|
||||
) else (
|
||||
powershell -Command "$r = try { (Invoke-WebRequest -Uri 'https://web.telegram.org' -TimeoutSec 10 -UseBasicParsing).StatusCode } catch { 0 }; if ($r -eq 200) { Write-Host ' [OK] web.telegram.org доступен!'; exit 0 } else { Write-Host ' [FAIL] web.telegram.org недоступен'; exit 1 }"
|
||||
if !errorlevel!==0 (
|
||||
set "best_mode=%%~m"
|
||||
echo [***] Рабочий режим найден: %%~m
|
||||
)
|
||||
)
|
||||
)
|
||||
|
||||
:: Пробуем кастомные режимы
|
||||
for %%p in (
|
||||
"-e 2 -f 2 -q -r -m"
|
||||
"-e 1 -f 1 -q -r -m -s"
|
||||
"-p -f 2 -e 2 -q"
|
||||
) do (
|
||||
set /a mode_idx+=1
|
||||
echo --- Тест !mode_idx!: GoodbyeDPI %%~p ---
|
||||
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
|
||||
start "" /B "%GDPI_EXE%" %%~p --blacklist "%BLACKLIST%"
|
||||
timeout /t 3 >nul
|
||||
|
||||
tasklist /fi "imagename eq goodbyedpi.exe" 2>nul | findstr /i "goodbyedpi" >nul
|
||||
if !errorlevel! neq 0 (
|
||||
echo [FAIL] Не запустился
|
||||
) else (
|
||||
powershell -Command "$r = try { (Invoke-WebRequest -Uri 'https://web.telegram.org' -TimeoutSec 10 -UseBasicParsing).StatusCode } catch { 0 }; if ($r -eq 200) { Write-Host ' [OK] web.telegram.org доступен!'; exit 0 } else { Write-Host ' [FAIL] web.telegram.org недоступен'; exit 1 }"
|
||||
if !errorlevel!==0 (
|
||||
if not defined best_mode (
|
||||
set "best_mode=%%~p"
|
||||
echo [***] Рабочий режим найден: %%~p
|
||||
)
|
||||
)
|
||||
)
|
||||
)
|
||||
|
||||
echo.
|
||||
if defined best_mode (
|
||||
echo ============================================================
|
||||
echo [OK] Лучший рабочий режим: !best_mode!
|
||||
echo Перезапускаю GoodbyeDPI с этим режимом...
|
||||
echo ============================================================
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" !best_mode! --blacklist "%BLACKLIST%"
|
||||
) else (
|
||||
echo ============================================================
|
||||
echo [!] Ни один режим GoodbyeDPI не помог.
|
||||
echo Возможно, нужен VPN или MTProxy.
|
||||
echo ============================================================
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
)
|
||||
echo.
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: 3. КОМБИНИРОВАННЫЙ РЕЖИМ
|
||||
:: ============================================================
|
||||
:COMBINED
|
||||
cls
|
||||
echo ============================================================
|
||||
echo Комбинированный режим (DNS + GoodbyeDPI)
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
:: --- DNS ---
|
||||
echo [*] Устанавливаю DNS Cloudflare (1.1.1.1)...
|
||||
for /f "tokens=*" %%a in ('powershell -Command "(Get-NetAdapter | Where-Object {$_.Status -eq 'Up'} | Select-Object -First 1).Name"') do (
|
||||
set "ADAPTER_NAME=%%a"
|
||||
)
|
||||
if defined ADAPTER_NAME (
|
||||
netsh interface ipv4 set dnsservers "!ADAPTER_NAME!" static 1.1.1.1 primary validate=no >nul 2>&1
|
||||
netsh interface ipv4 add dnsservers "!ADAPTER_NAME!" 1.0.0.1 index=2 validate=no >nul 2>&1
|
||||
ipconfig /flushdns >nul 2>&1
|
||||
echo [OK] DNS установлен: Cloudflare (1.1.1.1, 1.0.0.1)
|
||||
) else (
|
||||
echo [!] Не удалось определить сетевой адаптер для DNS.
|
||||
)
|
||||
|
||||
:: --- GoodbyeDPI ---
|
||||
if not exist "%GDPI_EXE%" (
|
||||
echo.
|
||||
echo [!] GoodbyeDPI не найден. Скачиваю...
|
||||
call :DO_DOWNLOAD_GDPI
|
||||
if not exist "%GDPI_EXE%" (
|
||||
echo [!] Не удалось скачать GoodbyeDPI. Работаю только с DNS.
|
||||
pause
|
||||
goto MENU
|
||||
)
|
||||
)
|
||||
|
||||
echo.
|
||||
echo [*] Запускаю GoodbyeDPI в режиме -9...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" -9 --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
|
||||
tasklist /fi "imagename eq goodbyedpi.exe" 2>nul | findstr /i "goodbyedpi" >nul
|
||||
if %errorlevel%==0 (
|
||||
echo [OK] GoodbyeDPI запущен.
|
||||
) else (
|
||||
echo [!] GoodbyeDPI не запустился. Пробую режим -5...
|
||||
start "" /B "%GDPI_EXE%" -5 --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
)
|
||||
|
||||
echo.
|
||||
echo ============================================================
|
||||
echo [OK] Комбинированный режим активирован:
|
||||
echo DNS: Cloudflare 1.1.1.1
|
||||
echo DPI: GoodbyeDPI
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
:: Быстрый тест
|
||||
echo [*] Проверяю доступность Telegram...
|
||||
powershell -Command "$r = try { $sw = [System.Diagnostics.Stopwatch]::StartNew(); $resp = Invoke-WebRequest -Uri 'https://web.telegram.org' -TimeoutSec 15 -UseBasicParsing; $sw.Stop(); Write-Host \"[OK] web.telegram.org доступен (${($sw.ElapsedMilliseconds)}ms)\"; } catch { Write-Host '[!] web.telegram.org пока недоступен, подождите немного...' }"
|
||||
echo.
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: 4. ТЕСТ СОЕДИНЕНИЯ
|
||||
:: ============================================================
|
||||
:TEST
|
||||
cls
|
||||
echo ============================================================
|
||||
echo Тест соединения с Telegram
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
echo [*] Проверяю GoodbyeDPI...
|
||||
tasklist /fi "imagename eq goodbyedpi.exe" 2>nul | findstr /i "goodbyedpi" >nul
|
||||
if %errorlevel%==0 (
|
||||
echo GoodbyeDPI: ЗАПУЩЕН
|
||||
) else (
|
||||
echo GoodbyeDPI: не запущен
|
||||
)
|
||||
echo.
|
||||
|
||||
echo [*] Текущий DNS:
|
||||
powershell -Command "Get-DnsClientServerAddress -AddressFamily IPv4 | Where-Object {$_.ServerAddresses.Count -gt 0} | Format-Table InterfaceAlias, ServerAddresses -AutoSize"
|
||||
echo.
|
||||
|
||||
echo [*] Пинг серверов Telegram (DC1-DC5)...
|
||||
echo.
|
||||
|
||||
set "TG_IPS=149.154.175.50 149.154.167.51 149.154.175.100 149.154.167.91 91.108.56.100 91.108.4.100"
|
||||
|
||||
for %%i in (%TG_IPS%) do (
|
||||
echo Пинг %%i...
|
||||
ping -n 1 -w 3000 %%i >nul 2>&1
|
||||
if !errorlevel!==0 (
|
||||
for /f "tokens=*" %%t in ('ping -n 1 -w 3000 %%i ^| findstr /i "time="') do (
|
||||
echo [OK] %%i — %%t
|
||||
)
|
||||
) else (
|
||||
echo [FAIL] %%i — недоступен
|
||||
)
|
||||
)
|
||||
|
||||
echo.
|
||||
echo [*] Проверяю TCP-соединение (порт 443)...
|
||||
echo.
|
||||
|
||||
for %%i in (149.154.175.50 149.154.167.51 91.108.56.100) do (
|
||||
powershell -Command "$tcp = New-Object System.Net.Sockets.TcpClient; try { $tcp.ConnectAsync('%%i', 443).Wait(5000) | Out-Null; if ($tcp.Connected) { Write-Host ' [OK] %%i:443 — TCP доступен' } else { Write-Host ' [FAIL] %%i:443 — таймаут' } } catch { Write-Host ' [FAIL] %%i:443 — отказано' } finally { $tcp.Dispose() }"
|
||||
)
|
||||
|
||||
echo.
|
||||
echo [*] Проверяю HTTPS доступность...
|
||||
echo.
|
||||
|
||||
for %%u in (
|
||||
"https://web.telegram.org"
|
||||
"https://core.telegram.org"
|
||||
"https://t.me"
|
||||
) do (
|
||||
powershell -Command "$url = '%%~u'; try { $sw = [System.Diagnostics.Stopwatch]::StartNew(); $r = Invoke-WebRequest -Uri $url -TimeoutSec 10 -UseBasicParsing; $sw.Stop(); $ms = $sw.ElapsedMilliseconds; Write-Host \" [OK] $url — HTTP $($r.StatusCode) (${ms}ms)\" } catch { Write-Host \" [FAIL] $url — недоступен\" }"
|
||||
)
|
||||
|
||||
echo.
|
||||
echo [*] Проверяю DNS резолвинг...
|
||||
echo.
|
||||
|
||||
for %%d in (web.telegram.org core.telegram.org t.me telegram.org) do (
|
||||
powershell -Command "try { $ip = [System.Net.Dns]::GetHostAddresses('%%d') | Select-Object -First 1; Write-Host \" [OK] %%d -> $($ip.IPAddressToString)\" } catch { Write-Host ' [FAIL] %%d — DNS не резолвится' }"
|
||||
)
|
||||
|
||||
echo.
|
||||
echo ============================================================
|
||||
echo Тест завершён
|
||||
echo ============================================================
|
||||
echo.
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: 5. СБРОС НАСТРОЕК
|
||||
:: ============================================================
|
||||
:RESET
|
||||
cls
|
||||
echo ============================================================
|
||||
echo Сброс настроек
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
echo [*] Останавливаю GoodbyeDPI...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
echo [OK] GoodbyeDPI остановлен.
|
||||
|
||||
echo.
|
||||
echo [*] Возвращаю DNS к автоматическому (DHCP)...
|
||||
for /f "tokens=*" %%a in ('powershell -Command "(Get-NetAdapter | Where-Object {$_.Status -eq 'Up'} | Select-Object -First 1).Name"') do (
|
||||
set "ADAPTER_NAME=%%a"
|
||||
)
|
||||
if defined ADAPTER_NAME (
|
||||
netsh interface ipv4 set dnsservers "!ADAPTER_NAME!" dhcp >nul 2>&1
|
||||
ipconfig /flushdns >nul 2>&1
|
||||
echo [OK] DNS сброшен на DHCP (автоматический).
|
||||
) else (
|
||||
echo [!] Не удалось определить адаптер. Сбросьте DNS вручную.
|
||||
)
|
||||
|
||||
echo.
|
||||
echo [OK] Все настройки сброшены.
|
||||
echo.
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: 6. ПОКАЗАТЬ ТЕКУЩИЕ НАСТРОЙКИ
|
||||
:: ============================================================
|
||||
:SHOW_NET
|
||||
cls
|
||||
echo ============================================================
|
||||
echo Текущие сетевые настройки
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
echo [*] Сетевые адаптеры:
|
||||
powershell -Command "Get-NetAdapter | Where-Object {$_.Status -eq 'Up'} | Format-Table Name, InterfaceDescription, Status, LinkSpeed -AutoSize"
|
||||
|
||||
echo [*] DNS серверы:
|
||||
powershell -Command "Get-DnsClientServerAddress -AddressFamily IPv4 | Where-Object {$_.ServerAddresses.Count -gt 0} | Format-Table InterfaceAlias, ServerAddresses -AutoSize"
|
||||
|
||||
echo [*] IP-конфигурация:
|
||||
ipconfig | findstr /i "IPv4 Subnet Gateway DNS"
|
||||
|
||||
echo.
|
||||
echo [*] GoodbyeDPI:
|
||||
tasklist /fi "imagename eq goodbyedpi.exe" 2>nul | findstr /i "goodbyedpi" >nul
|
||||
if %errorlevel%==0 (
|
||||
echo Статус: ЗАПУЩЕН
|
||||
for /f "tokens=2" %%p in ('tasklist /fi "imagename eq goodbyedpi.exe" /fo list ^| findstr "PID"') do (
|
||||
echo PID: %%p
|
||||
)
|
||||
) else (
|
||||
echo Статус: не запущен
|
||||
)
|
||||
|
||||
echo.
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: СКАЧИВАНИЕ GoodbyeDPI
|
||||
:: ============================================================
|
||||
:DOWNLOAD_GDPI
|
||||
echo.
|
||||
echo [*] GoodbyeDPI необходим для обхода DPI.
|
||||
echo Скачать автоматически с GitHub?
|
||||
echo.
|
||||
echo [1] Да, скачать
|
||||
echo [0] Нет, назад
|
||||
echo.
|
||||
set /p "dl_choice=Выбор [0-1]: "
|
||||
if "%dl_choice%"=="0" goto MENU
|
||||
if "%dl_choice%"=="1" (
|
||||
call :DO_DOWNLOAD_GDPI
|
||||
if exist "%GDPI_EXE%" goto DPI_MODE_MENU
|
||||
pause
|
||||
goto MENU
|
||||
)
|
||||
goto MENU
|
||||
|
||||
:DO_DOWNLOAD_GDPI
|
||||
echo.
|
||||
echo [*] Создаю папку tools...
|
||||
if not exist "%TOOLS_DIR%" mkdir "%TOOLS_DIR%"
|
||||
|
||||
echo [*] Скачиваю GoodbyeDPI с GitHub...
|
||||
echo URL: %GDPI_URL%
|
||||
echo.
|
||||
|
||||
powershell -Command "[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; try { Invoke-WebRequest -Uri '%GDPI_URL%' -OutFile '%GDPI_ZIP%' -UseBasicParsing; Write-Host '[OK] Скачано успешно' } catch { Write-Host \"[!] Ошибка скачивания: $($_.Exception.Message)\"; exit 1 }"
|
||||
|
||||
if not exist "%GDPI_ZIP%" (
|
||||
echo [!] Не удалось скачать файл.
|
||||
echo Попробуйте скачать вручную:
|
||||
echo %GDPI_URL%
|
||||
echo и распаковать в %GDPI_DIR%
|
||||
exit /b 1
|
||||
)
|
||||
|
||||
echo [*] Распаковываю...
|
||||
powershell -Command "try { Expand-Archive -Path '%GDPI_ZIP%' -DestinationPath '%TOOLS_DIR%' -Force; Write-Host '[OK] Распаковано' } catch { Write-Host \"[!] Ошибка распаковки: $($_.Exception.Message)\"; exit 1 }"
|
||||
|
||||
:: Ищем goodbyedpi.exe рекурсивно
|
||||
if not exist "%GDPI_EXE%" (
|
||||
echo [*] Ищу goodbyedpi.exe...
|
||||
for /f "tokens=*" %%f in ('powershell -Command "Get-ChildItem -Path '%TOOLS_DIR%' -Recurse -Filter 'goodbyedpi.exe' | Select-Object -First 1 | ForEach-Object { $_.DirectoryName }"') do (
|
||||
set "FOUND_DIR=%%f"
|
||||
)
|
||||
if defined FOUND_DIR (
|
||||
echo [*] Найден в: !FOUND_DIR!
|
||||
set "GDPI_DIR=!FOUND_DIR!"
|
||||
set "GDPI_EXE=!FOUND_DIR!\goodbyedpi.exe"
|
||||
|
||||
:: Обновим путь для x86_64
|
||||
if not exist "!FOUND_DIR!\goodbyedpi.exe" (
|
||||
for /f "tokens=*" %%f in ('powershell -Command "Get-ChildItem -Path '%TOOLS_DIR%' -Recurse -Filter 'goodbyedpi.exe' | Select-Object -First 1 | ForEach-Object { $_.FullName }"') do (
|
||||
set "GDPI_EXE=%%f"
|
||||
echo [*] Путь к exe: %%f
|
||||
)
|
||||
)
|
||||
)
|
||||
)
|
||||
|
||||
if exist "%GDPI_EXE%" (
|
||||
echo.
|
||||
echo [OK] GoodbyeDPI установлен: %GDPI_EXE%
|
||||
del "%GDPI_ZIP%" >nul 2>&1
|
||||
) else (
|
||||
echo.
|
||||
echo [!] goodbyedpi.exe не найден после распаковки.
|
||||
echo Проверьте папку: %TOOLS_DIR%
|
||||
)
|
||||
exit /b 0
|
||||
|
||||
:: ============================================================
|
||||
:: 7. АВТО-ОБХОД (одна кнопка) — замеряет ВСЕ режимы,
|
||||
:: выбирает самый быстрый
|
||||
:: ============================================================
|
||||
:AUTO_BYPASS
|
||||
cls
|
||||
echo ============================================================
|
||||
echo АВТО-ОБХОД — тестирую ВСЕ режимы, выбираю лучший...
|
||||
echo Это займёт 2-3 минуты.
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
:: --- Шаг 1: Определяю адаптер ---
|
||||
echo [1] Определяю сетевой адаптер...
|
||||
set "ADAPTER_NAME="
|
||||
for /f "tokens=*" %%a in ('powershell -Command "(Get-NetAdapter | Where-Object {$_.Status -eq 'Up'} | Select-Object -First 1).Name"') do (
|
||||
set "ADAPTER_NAME=%%a"
|
||||
)
|
||||
if not defined ADAPTER_NAME (
|
||||
echo [!] Не удалось определить адаптер.
|
||||
pause
|
||||
goto MENU
|
||||
)
|
||||
echo Адаптер: !ADAPTER_NAME!
|
||||
|
||||
:: --- Шаг 2: DNS ---
|
||||
echo.
|
||||
echo [2] Устанавливаю DNS Cloudflare (1.1.1.1)...
|
||||
netsh interface ipv4 set dnsservers "!ADAPTER_NAME!" static 1.1.1.1 primary validate=no >nul 2>&1
|
||||
netsh interface ipv4 add dnsservers "!ADAPTER_NAME!" 1.0.0.1 index=2 validate=no >nul 2>&1
|
||||
ipconfig /flushdns >nul 2>&1
|
||||
echo OK
|
||||
|
||||
:: --- Шаг 3: GoodbyeDPI ---
|
||||
echo.
|
||||
echo [3] Ищу GoodbyeDPI...
|
||||
if not exist "%GDPI_EXE%" (
|
||||
echo Не найден, скачиваю...
|
||||
call :DO_DOWNLOAD_GDPI
|
||||
if not exist "%GDPI_EXE%" (
|
||||
echo [!] Не удалось скачать GoodbyeDPI.
|
||||
pause
|
||||
goto MENU
|
||||
)
|
||||
)
|
||||
echo OK: %GDPI_EXE%
|
||||
|
||||
:: --- Шаг 4: Бенчмарк каждого режима ---
|
||||
echo.
|
||||
echo ============================================================
|
||||
echo Замеряю скорость каждого режима...
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
set "BEST_MODE="
|
||||
set "BEST_SCORE=99999"
|
||||
set "BEST_LABEL="
|
||||
set "MODE_NUM=0"
|
||||
|
||||
:: Benchmark function is a powershell one-liner that does
|
||||
:: 2x TCP + 1x HTTPS, returns average ms (lower=better), 99999 if fail
|
||||
set "BENCH_CMD=powershell -Command "$total=0; $ok=0; foreach($ip in @('149.154.167.51','149.154.175.50','91.108.56.100')) { foreach($i in 1..2) { $tcp=New-Object Net.Sockets.TcpClient; $sw=[Diagnostics.Stopwatch]::StartNew(); try { [void]$tcp.ConnectAsync($ip,443).Wait(4000); if($tcp.Connected){$sw.Stop();$total+=$sw.ElapsedMilliseconds;$ok++} } catch {} finally {$tcp.Dispose()} } }; try { $sw2=[Diagnostics.Stopwatch]::StartNew(); $r=Invoke-WebRequest 'https://web.telegram.org' -TimeoutSec 8 -UseBasicParsing; $sw2.Stop(); $total+=$sw2.ElapsedMilliseconds*3; $ok+=3 } catch {}; if($ok -eq 0){Write-Host 99999}else{Write-Host([math]::Floor($total/$ok))}""
|
||||
|
||||
:: Mode 1: без GoodbyeDPI (только DNS)
|
||||
set /a MODE_NUM+=1
|
||||
echo [%MODE_NUM%/8] Только DNS (без GoodbyeDPI)...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
for /f "tokens=*" %%s in ('%BENCH_CMD%') do set "SCORE=%%s"
|
||||
echo Скорость: !SCORE!ms
|
||||
if !SCORE! LSS !BEST_SCORE! (
|
||||
set "BEST_SCORE=!SCORE!"
|
||||
set "BEST_MODE=dns_only"
|
||||
set "BEST_LABEL=Только DNS"
|
||||
)
|
||||
|
||||
:: Mode 2: -9
|
||||
set /a MODE_NUM+=1
|
||||
echo [%MODE_NUM%/8] GoodbyeDPI -9 (максимальная защита)...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" -9 --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
for /f "tokens=*" %%s in ('%BENCH_CMD%') do set "SCORE=%%s"
|
||||
echo Скорость: !SCORE!ms
|
||||
if !SCORE! LSS !BEST_SCORE! (
|
||||
set "BEST_SCORE=!SCORE!"
|
||||
set "BEST_MODE=-9"
|
||||
set "BEST_LABEL=GoodbyeDPI -9"
|
||||
)
|
||||
|
||||
:: Mode 3: -5
|
||||
set /a MODE_NUM+=1
|
||||
echo [%MODE_NUM%/8] GoodbyeDPI -5 (средний)...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" -5 --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
for /f "tokens=*" %%s in ('%BENCH_CMD%') do set "SCORE=%%s"
|
||||
echo Скорость: !SCORE!ms
|
||||
if !SCORE! LSS !BEST_SCORE! (
|
||||
set "BEST_SCORE=!SCORE!"
|
||||
set "BEST_MODE=-5"
|
||||
set "BEST_LABEL=GoodbyeDPI -5"
|
||||
)
|
||||
|
||||
:: Mode 4: -1
|
||||
set /a MODE_NUM+=1
|
||||
echo [%MODE_NUM%/8] GoodbyeDPI -1 (базовый)...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" -1 --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
for /f "tokens=*" %%s in ('%BENCH_CMD%') do set "SCORE=%%s"
|
||||
echo Скорость: !SCORE!ms
|
||||
if !SCORE! LSS !BEST_SCORE! (
|
||||
set "BEST_SCORE=!SCORE!"
|
||||
set "BEST_MODE=-1"
|
||||
set "BEST_LABEL=GoodbyeDPI -1"
|
||||
)
|
||||
|
||||
:: Mode 5: custom -e2 -f2 -q -r -m
|
||||
set /a MODE_NUM+=1
|
||||
echo [%MODE_NUM%/8] GoodbyeDPI -e2 -f2 -q -r -m...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" -e 2 -f 2 -q -r -m --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
for /f "tokens=*" %%s in ('%BENCH_CMD%') do set "SCORE=%%s"
|
||||
echo Скорость: !SCORE!ms
|
||||
if !SCORE! LSS !BEST_SCORE! (
|
||||
set "BEST_SCORE=!SCORE!"
|
||||
set "BEST_MODE=-e 2 -f 2 -q -r -m"
|
||||
set "BEST_LABEL=GoodbyeDPI -e2 -f2 -q -r -m"
|
||||
)
|
||||
|
||||
:: Mode 6: custom -p -f2 -e2 -q
|
||||
set /a MODE_NUM+=1
|
||||
echo [%MODE_NUM%/8] GoodbyeDPI -p -f2 -e2 -q...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" -p -f 2 -e 2 -q --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
for /f "tokens=*" %%s in ('%BENCH_CMD%') do set "SCORE=%%s"
|
||||
echo Скорость: !SCORE!ms
|
||||
if !SCORE! LSS !BEST_SCORE! (
|
||||
set "BEST_SCORE=!SCORE!"
|
||||
set "BEST_MODE=-p -f 2 -e 2 -q"
|
||||
set "BEST_LABEL=GoodbyeDPI -p -f2 -e2 -q"
|
||||
)
|
||||
|
||||
:: Mode 7: custom -e40 -f2 -q -r -m
|
||||
set /a MODE_NUM+=1
|
||||
echo [%MODE_NUM%/8] GoodbyeDPI -e40 -f2 -q -r -m...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" -e 40 -f 2 -q -r -m --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
for /f "tokens=*" %%s in ('%BENCH_CMD%') do set "SCORE=%%s"
|
||||
echo Скорость: !SCORE!ms
|
||||
if !SCORE! LSS !BEST_SCORE! (
|
||||
set "BEST_SCORE=!SCORE!"
|
||||
set "BEST_MODE=-e 40 -f 2 -q -r -m"
|
||||
set "BEST_LABEL=GoodbyeDPI -e40 -f2 -q -r -m"
|
||||
)
|
||||
|
||||
:: Mode 8: -9 --set-ttl 5
|
||||
set /a MODE_NUM+=1
|
||||
echo [%MODE_NUM%/8] GoodbyeDPI -9 --set-ttl 5...
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
start "" /B "%GDPI_EXE%" -9 --set-ttl 5 --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
for /f "tokens=*" %%s in ('%BENCH_CMD%') do set "SCORE=%%s"
|
||||
echo Скорость: !SCORE!ms
|
||||
if !SCORE! LSS !BEST_SCORE! (
|
||||
set "BEST_SCORE=!SCORE!"
|
||||
set "BEST_MODE=-9 --set-ttl 5"
|
||||
set "BEST_LABEL=GoodbyeDPI -9 --set-ttl 5"
|
||||
)
|
||||
|
||||
:: --- Применяем лучший ---
|
||||
echo.
|
||||
echo ============================================================
|
||||
|
||||
if "!BEST_SCORE!"=="99999" (
|
||||
echo [!] Ни один режим не сработал. Попробуйте VPN или MTProxy.
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
pause
|
||||
goto MENU
|
||||
)
|
||||
|
||||
echo ЛУЧШИЙ РЕЖИМ: !BEST_LABEL!
|
||||
echo СКОРОСТЬ: !BEST_SCORE!ms
|
||||
echo ============================================================
|
||||
echo.
|
||||
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
timeout /t 1 >nul
|
||||
|
||||
if "!BEST_MODE!"=="dns_only" (
|
||||
echo [OK] DNS Cloudflare достаточно, GoodbyeDPI не нужен.
|
||||
) else (
|
||||
echo [*] Запускаю GoodbyeDPI с лучшими параметрами...
|
||||
start "" /B "%GDPI_EXE%" !BEST_MODE! --blacklist "%BLACKLIST%"
|
||||
timeout /t 2 >nul
|
||||
echo [OK] GoodbyeDPI запущен: !BEST_MODE!
|
||||
)
|
||||
|
||||
echo.
|
||||
echo ============================================================
|
||||
echo [OK] ОБХОД АКТИВИРОВАН!
|
||||
echo DNS: Cloudflare 1.1.1.1
|
||||
echo Режим: !BEST_LABEL! (!BEST_SCORE!ms)
|
||||
echo Для остановки — пункт 5 в меню.
|
||||
echo ============================================================
|
||||
pause
|
||||
goto MENU
|
||||
|
||||
:: ============================================================
|
||||
:: ВЫХОД
|
||||
:: ============================================================
|
||||
:EXIT
|
||||
echo.
|
||||
echo [*] GoodbyeDPI будет остановлен при выходе? (y/n)
|
||||
set /p "exit_choice="
|
||||
if /i "%exit_choice%"=="y" (
|
||||
taskkill /f /im goodbyedpi.exe >nul 2>&1
|
||||
echo [OK] GoodbyeDPI остановлен.
|
||||
)
|
||||
echo.
|
||||
echo Bye!
|
||||
endlocal
|
||||
exit /b 0
|
||||
418
РЕШЕНИЯ.md
418
РЕШЕНИЯ.md
@@ -1,418 +0,0 @@
|
||||
# Решения задач по теории вероятностей
|
||||
## Формулы Бернулли, местная и интегральная теоремы Муавра–Лапласа, формула Пуассона
|
||||
|
||||
---
|
||||
|
||||
## Задача 3
|
||||
|
||||
**Условие:** В среднем каждое второе малое предприятие имеет нарушение финансовой дисциплины. Из **n = 1000** предприятий найти вероятность того, что нарушения имеют:
|
||||
|
||||
Дано: n = 1000, p = 1/2 = 0,5, q = 0,5
|
||||
npq = 1000 · 0,5 · 0,5 = 250, √(npq) = √250 ≈ 15,81
|
||||
|
||||
---
|
||||
|
||||
### а) Ровно 480 предприятий
|
||||
|
||||
Используем **местную теорему Муавра–Лапласа**:
|
||||
|
||||
$$P_n(k) \approx \frac{1}{\sqrt{npq}} \cdot \varphi(x), \quad x = \frac{k - np}{\sqrt{npq}}$$
|
||||
|
||||
$$x = \frac{480 - 500}{15{,}81} = \frac{-20}{15{,}81} \approx -1{,}26$$
|
||||
|
||||
φ(1,26) = 0,1826 (по таблице функции Гаусса)
|
||||
|
||||
$$P_{1000}(480) \approx \frac{0{,}1826}{15{,}81} \approx \mathbf{0{,}0116}$$
|
||||
|
||||
---
|
||||
|
||||
### б) Наивероятнейшее число предприятий
|
||||
|
||||
По формуле: **np − q ≤ k₀ ≤ np + p**
|
||||
|
||||
499,5 ≤ k₀ ≤ 500,5 → **k₀ = 500**
|
||||
|
||||
---
|
||||
|
||||
### в) Не менее 480 предприятий
|
||||
|
||||
Используем **интегральную теорему Муавра–Лапласа**:
|
||||
|
||||
$$P(k \geq 480) = 0{,}5 + \Phi\!\left(\frac{np - k_1}{\sqrt{npq}}\right)$$
|
||||
|
||||
$$= 0{,}5 + \Phi(1{,}26) = 0{,}5 + 0{,}3962 \approx \mathbf{0{,}8962}$$
|
||||
|
||||
---
|
||||
|
||||
### г) От 480 до 520 предприятий
|
||||
|
||||
$$x_1 = \frac{480 - 500}{15{,}81} \approx -1{,}26, \quad x_2 = \frac{520 - 500}{15{,}81} \approx 1{,}26$$
|
||||
|
||||
$$P(480 \leq k \leq 520) = \Phi(1{,}26) - \Phi(-1{,}26) = 2\Phi(1{,}26) = 2 \cdot 0{,}3962 \approx \mathbf{0{,}7924}$$
|
||||
|
||||
---
|
||||
|
||||
## Задача 4
|
||||
|
||||
**Условие:** Вероятность банкротства предприятия за время t равна 0,2. Из **n = 6** предприятий найти вероятность того, что **сохранятся**:
|
||||
|
||||
Дано: n = 6, p(сохранится) = 1 − 0,2 = **0,8**, q = 0,2
|
||||
|
||||
Используем **формулу Бернулли**: P_n(k) = C(n,k) · pᵏ · qⁿ⁻ᵏ
|
||||
|
||||
---
|
||||
|
||||
### а) Два предприятия (k = 2)
|
||||
|
||||
$$P_6(2) = C_6^2 \cdot (0{,}8)^2 \cdot (0{,}2)^4 = 15 \cdot 0{,}64 \cdot 0{,}0016 = \mathbf{0{,}0154}$$
|
||||
|
||||
---
|
||||
|
||||
### б) Более двух предприятий (k > 2)
|
||||
|
||||
$$P(k > 2) = 1 - P(k \leq 2) = 1 - [P(0) + P(1) + P(2)]$$
|
||||
|
||||
- P₆(0) = C(6,0) · (0,8)⁰ · (0,2)⁶ = 1 · 1 · 0,000064 = 0,000064
|
||||
- P₆(1) = C(6,1) · (0,8)¹ · (0,2)⁵ = 6 · 0,8 · 0,00032 = 0,001536
|
||||
- P₆(2) = 0,0154 (вычислено выше)
|
||||
|
||||
$$P(k > 2) = 1 - (0{,}000064 + 0{,}001536 + 0{,}0154) \approx 1 - 0{,}0170 = \mathbf{0{,}983}$$
|
||||
|
||||
---
|
||||
|
||||
## Задача 5
|
||||
|
||||
**Условие:** В банк отправлено **n = 4000** пакетов. Вероятность ошибки в комплектации **p = 0,0001**. Найти вероятность обнаружения:
|
||||
|
||||
λ = np = 4000 · 0,0001 = **0,4**
|
||||
|
||||
Используем **формулу Пуассона**: P(k) = (λᵏ / k!) · e⁻λ
|
||||
|
||||
e⁻⁰·⁴ ≈ 0,6703
|
||||
|
||||
---
|
||||
|
||||
### а) Три ошибочно укомплектованных пакета (k = 3)
|
||||
|
||||
$$P(3) = \frac{(0{,}4)^3}{3!} \cdot e^{-0{,}4} = \frac{0{,}064}{6} \cdot 0{,}6703 \approx \mathbf{0{,}00715}$$
|
||||
|
||||
---
|
||||
|
||||
### б) Не более трёх пакетов (k ≤ 3)
|
||||
|
||||
$$P(k \leq 3) = P(0) + P(1) + P(2) + P(3)$$
|
||||
|
||||
- P(0) = e⁻⁰·⁴ = 0,6703
|
||||
- P(1) = 0,4 · e⁻⁰·⁴ = 0,2681
|
||||
- P(2) = (0,4²/2) · e⁻⁰·⁴ = 0,0536
|
||||
- P(3) = 0,00715
|
||||
|
||||
$$P(k \leq 3) = 0{,}6703 + 0{,}2681 + 0{,}0536 + 0{,}00715 \approx \mathbf{0{,}9992}$$
|
||||
|
||||
---
|
||||
|
||||
---
|
||||
|
||||
## Домашняя работа
|
||||
|
||||
---
|
||||
|
||||
## Задача 1 (ДЗ)
|
||||
|
||||
**Условие:** Вероятность попадания в цель р = 0,7. Из **n = 80** выстрелов — найти вероятность различных исходов.
|
||||
|
||||
Дано: n = 80, p = 0,7, q = 0,3
|
||||
np = 56, npq = 16,8, √(npq) ≈ 4,10
|
||||
|
||||
---
|
||||
|
||||
### а) Ровно 75 попаданий
|
||||
|
||||
$$x = \frac{75 - 56}{4{,}10} \approx 4{,}63$$
|
||||
|
||||
φ(4,63) ≈ 0,00002 (крайне малое значение)
|
||||
|
||||
$$P_{80}(75) \approx \frac{0{,}00002}{4{,}10} \approx \mathbf{0{,}000005} \approx 0$$
|
||||
|
||||
---
|
||||
|
||||
### б) Не менее 75 попаданий
|
||||
|
||||
$$x_1 = \frac{75 - 56}{4{,}10} \approx 4{,}63$$
|
||||
|
||||
$$P(k \geq 75) = 0{,}5 - \Phi(4{,}63) \approx 0{,}5 - 0{,}5 \approx \mathbf{0}$$
|
||||
|
||||
---
|
||||
|
||||
### в) Менее 75 попаданий
|
||||
|
||||
$$P(k < 75) = 1 - P(k \geq 75) \approx \mathbf{1}$$
|
||||
|
||||
---
|
||||
|
||||
### г) Не более 75 попаданий
|
||||
|
||||
$$P(k \leq 75) = P(k < 75) + P(k = 75) \approx 1 + 0 \approx \mathbf{1}$$
|
||||
|
||||
---
|
||||
|
||||
### д) Более 75 попаданий
|
||||
|
||||
$$P(k > 75) \approx \mathbf{0}$$
|
||||
|
||||
---
|
||||
|
||||
### е) Все 80 выстрелов
|
||||
|
||||
$$P_{80}(80) = (0{,}7)^{80} = e^{80 \ln 0{,}7} = e^{-28{,}5} \approx \mathbf{4{,}4 \times 10^{-13}}$$
|
||||
|
||||
---
|
||||
|
||||
## Задача 2 (ДЗ)
|
||||
|
||||
**Условие:** Вероятность выпуска сверла с браком p = 0,02. Сверла укладываются по 100 штук. Найти наименьшее количество добавочных сверл, чтобы с вероятностью **не менее 0,9** в коробке было **не менее 100 исправных**.
|
||||
|
||||
Пусть в коробке **n** сверл, p(исправное) = 0,98. Нужно: P(X ≥ 100) ≥ 0,9
|
||||
|
||||
$$P(X \geq 100) = 0{,}5 + \Phi\!\left(\frac{np - 100}{\sqrt{np \cdot 0{,}02}}\right) \geq 0{,}9$$
|
||||
|
||||
$$\Rightarrow \frac{0{,}98n - 100}{0{,}14\sqrt{n}} \geq 1{,}28$$
|
||||
|
||||
Подстановка t = √n: 0,98t² − 0,1792t − 100 ≥ 0
|
||||
|
||||
$$t = \frac{0{,}1792 + \sqrt{0{,}1792^2 + 4 \cdot 0{,}98 \cdot 100}}{2 \cdot 0{,}98} = \frac{0{,}179 + 19{,}80}{1{,}96} \approx 10{,}19$$
|
||||
|
||||
$$n \geq (10{,}19)^2 \approx 103{,}9 \Rightarrow n = 104$$
|
||||
|
||||
**Ответ:** нужно добавить **m = 104 − 100 = 4 сверла**
|
||||
|
||||
---
|
||||
|
||||
## Задача 3 (ДЗ)
|
||||
|
||||
**Условие:** Сколько изюминок в среднем должна содержать булочка, чтобы вероятность хотя бы одной изюминки была **не менее 0,99**?
|
||||
|
||||
Используем формулу Пуассона: P(X ≥ 1) = 1 − P(X = 0) = 1 − e⁻λ ≥ 0,99
|
||||
|
||||
$$e^{-\lambda} \leq 0{,}01 \Rightarrow \lambda \geq -\ln(0{,}01) = \ln 100 \approx 4{,}61$$
|
||||
|
||||
**Ответ:** в среднем в булочке должно быть **не менее 5 изюминок** (λ ≥ 4,61)
|
||||
|
||||
---
|
||||
|
||||
---
|
||||
|
||||
## Задача (Картофель)
|
||||
|
||||
**Условие:** В складе 20% клубней с пятнами. Отобрано **n = 9** клубней. p(без пятен) = 0,8, q = 0,2.
|
||||
|
||||
---
|
||||
|
||||
### а) Наивероятнейшее число клубней без пятен
|
||||
|
||||
np − q ≤ k₀ ≤ np + p
|
||||
9·0,8 − 0,2 ≤ k₀ ≤ 9·0,8 + 0,8
|
||||
**7,0 ≤ k₀ ≤ 8,0**
|
||||
|
||||
Так как оба конца — целые числа, оба значения k₀ = 7 и k₀ = 8 являются наивероятнейшими (с одинаковой вероятностью).
|
||||
|
||||
---
|
||||
|
||||
### б) Вероятность наивероятнейшего числа
|
||||
|
||||
$$P_9(7) = C_9^7 \cdot (0{,}8)^7 \cdot (0{,}2)^2 = 36 \cdot 0{,}2097 \cdot 0{,}04 \approx \mathbf{0{,}302}$$
|
||||
|
||||
$$P_9(8) = C_9^8 \cdot (0{,}8)^8 \cdot (0{,}2)^1 = 9 \cdot 0{,}1678 \cdot 0{,}2 \approx \mathbf{0{,}302}$$
|
||||
|
||||
P(7) = P(8) ≈ **0,302** — подтверждает, что оба значения равновероятны
|
||||
|
||||
---
|
||||
|
||||
## Задача (Событие А, k₀ = 20)
|
||||
|
||||
**Условие:** Вероятность события А в каждом испытании p = 0,7. Сколько испытаний нужно провести, чтобы **наивероятнейшее число** равнялось **20**?
|
||||
|
||||
np − q ≤ k₀ ≤ np + p, то есть при k₀ = 20:
|
||||
|
||||
$$0{,}7n - 0{,}3 \leq 20 \leq 0{,}7n + 0{,}7$$
|
||||
|
||||
Из левого неравенства: 0,7n ≤ 20,3 → **n ≤ 29**
|
||||
|
||||
Из правого неравенства: 0,7n ≥ 19,3 → **n ≥ 27,57 → n ≥ 28**
|
||||
|
||||
Проверка:
|
||||
- n = 28: np−q = 19,3, np+p = 20,3 → единственное целое k₀ = **20** ✓
|
||||
- n = 29: np−q = 20,0, np+p = 21,0 → k₀ = 20 или 21 (два наивероятнейших)
|
||||
|
||||
**Ответ: n = 28** (или n = 29, если допустимо два наивероятнейших значения)
|
||||
|
||||
---
|
||||
|
||||
---
|
||||
|
||||
## Задача 1 (Лотерея)
|
||||
|
||||
**Условие:** Вероятность выигрыша по одному билету p = 1/7. Имея **n = 7** билетов, найти вероятность выигрыша:
|
||||
|
||||
Используем **формулу Бернулли**
|
||||
|
||||
---
|
||||
|
||||
### а) По двум билетам (k = 2)
|
||||
|
||||
$$P_7(2) = C_7^2 \cdot \left(\frac{1}{7}\right)^2 \cdot \left(\frac{6}{7}\right)^5 = 21 \cdot \frac{1}{49} \cdot \frac{7776}{16807} = \frac{163296}{823543} \approx \mathbf{0{,}198}$$
|
||||
|
||||
---
|
||||
|
||||
### б) По трём билетам (k = 3)
|
||||
|
||||
$$P_7(3) = C_7^3 \cdot \left(\frac{1}{7}\right)^3 \cdot \left(\frac{6}{7}\right)^4 = 35 \cdot \frac{1}{343} \cdot \frac{1296}{2401} = \frac{45360}{823543} \approx \mathbf{0{,}055}$$
|
||||
|
||||
---
|
||||
|
||||
## Задача 2 (Мята, гербициды)
|
||||
|
||||
**Условие:** Повреждены гербицидами 15% растений мяты. Отобрано **n = 20** растений. p = 0,15, q = 0,85.
|
||||
|
||||
np − q ≤ k₀ ≤ np + p
|
||||
3 − 0,85 ≤ k₀ ≤ 3 + 0,15
|
||||
2,15 ≤ k₀ ≤ 3,15
|
||||
|
||||
**k₀ = 3** — наивероятнейшее число повреждённых растений
|
||||
|
||||
---
|
||||
|
||||
## Задача 3 (Сбербанк)
|
||||
|
||||
**Условие:** Два филиала. Филиал 1: n₁ = 120, p₁ = 0,94. Филиал 2: n₂ = 140, p₂ = 0,8. Найти наивероятнейшее число клиентов, снявших деньги.
|
||||
|
||||
**Филиал 1:** np₁ = 112,8; np₁ − q₁ = 112,74; np₁ + p₁ = 113,74 → **k₀₁ = 113**
|
||||
|
||||
**Филиал 2:** np₂ = 112,0; np₂ − q₂ = 111,8; np₂ + p₂ = 112,8 → **k₀₂ = 112**
|
||||
|
||||
**Ответ:** Первый филиал обслуживает больше клиентов, снявших деньги: **113 > 112**
|
||||
|
||||
---
|
||||
|
||||
---
|
||||
|
||||
## Задача (Событие А, 400 испытаний)
|
||||
|
||||
**Условие:** Найти вероятность того, что событие А наступит **ровно 80 раз** в 400 испытаниях, если p = 0,2.
|
||||
|
||||
Дано: n = 400, k = 80, p = 0,2, q = 0,8
|
||||
np = 80, npq = 64, √(npq) = 8
|
||||
|
||||
$$x = \frac{k - np}{\sqrt{npq}} = \frac{80 - 80}{8} = 0$$
|
||||
|
||||
$$\varphi(0) = \frac{1}{\sqrt{2\pi}} \approx 0{,}3989$$
|
||||
|
||||
$$P_{400}(80) \approx \frac{\varphi(0)}{\sqrt{npq}} = \frac{0{,}3989}{8} \approx \mathbf{0{,}0499}$$
|
||||
|
||||
---
|
||||
|
||||
## Задача (Стрелок)
|
||||
|
||||
**Условие:** Вероятность поражения мишени p = 0,75. При **n = 10** выстрелах найти вероятность **ровно 8 попаданий**.
|
||||
|
||||
$$P_{10}(8) = C_{10}^8 \cdot (0{,}75)^8 \cdot (0{,}25)^2$$
|
||||
|
||||
$$= 45 \cdot \frac{6561}{65536} \cdot \frac{1}{16} = \frac{295245}{1048576} \approx \mathbf{0{,}2816}$$
|
||||
|
||||
---
|
||||
|
||||
---
|
||||
|
||||
## Задача (Кинескопы — Интегральная теорема Лапласа)
|
||||
|
||||
**Условие:** 12% кинескопов не проработают гарантийный срок. Из **n = 50** наугад выбранных — найти вероятность того, что проработают гарантийный срок:
|
||||
|
||||
Дано: n = 50, p(проработает) = 1 − 0,12 = **0,88**, q = 0,12
|
||||
np = 44, npq = 50 · 0,88 · 0,12 = 5,28, √(npq) ≈ 2,298
|
||||
|
||||
---
|
||||
|
||||
### а) Ровно 47 кинескопов
|
||||
|
||||
Местная теорема:
|
||||
|
||||
$$x = \frac{47 - 44}{2{,}298} \approx 1{,}305, \quad \varphi(1{,}305) \approx 0{,}1714$$
|
||||
|
||||
$$P_{50}(47) \approx \frac{0{,}1714}{2{,}298} \approx \mathbf{0{,}0745}$$
|
||||
|
||||
---
|
||||
|
||||
### б) Не менее 47 кинескопов
|
||||
|
||||
$$x_1 = \frac{47 - 44}{2{,}298} \approx 1{,}305$$
|
||||
|
||||
$$P(k \geq 47) \approx 0{,}5 - \Phi(1{,}305) = 0{,}5 - 0{,}4040 \approx \mathbf{0{,}0960}$$
|
||||
|
||||
---
|
||||
|
||||
### в) Менее 47 кинескопов
|
||||
|
||||
$$P(k < 47) = 1 - P(k \geq 47) \approx 1 - 0{,}0960 = \mathbf{0{,}9040}$$
|
||||
|
||||
---
|
||||
|
||||
### г) Более 47 кинескопов
|
||||
|
||||
$$x = \frac{48 - 44}{2{,}298} \approx 1{,}740$$
|
||||
|
||||
$$P(k > 47) = P(k \geq 48) \approx 0{,}5 - \Phi(1{,}74) = 0{,}5 - 0{,}4591 \approx \mathbf{0{,}0409}$$
|
||||
|
||||
---
|
||||
|
||||
### д) Не более 47 кинескопов
|
||||
|
||||
$$P(k \leq 47) = 1 - P(k > 47) \approx 1 - 0{,}0409 = \mathbf{0{,}9591}$$
|
||||
|
||||
---
|
||||
|
||||
### е) Все 50 кинескопов
|
||||
|
||||
$$P_{50}(50) = (0{,}88)^{50} \approx \mathbf{0{,}00167}$$
|
||||
|
||||
(Для проверки по местной теореме: x = (50−44)/2,298 ≈ 2,61; φ(2,61) ≈ 0,0136; P ≈ 0,006 — приближение менее точно из-за удалённости от центра)
|
||||
|
||||
---
|
||||
|
||||
---
|
||||
|
||||
## Домашнее задание (Интегральная теорема Лапласа)
|
||||
|
||||
---
|
||||
|
||||
## Задача 1 (ДЗ — Новорождённые)
|
||||
|
||||
**Условие:** Среди **n = 1000** новорождённых. Вероятность рождения мальчика p = 0,51. Найти вероятность того, что мальчиков будет:
|
||||
|
||||
np = 510, npq = 1000 · 0,51 · 0,49 = 249,9, √(npq) ≈ 15,81
|
||||
|
||||
---
|
||||
|
||||
### а) Не менее половины (m ≥ 500)
|
||||
|
||||
$$x_1 = \frac{500 - 510}{15{,}81} \approx -0{,}63, \quad x_2 \to +\infty$$
|
||||
|
||||
$$P_{1000}(500 \leq m \leq 1000) \approx \Phi(+\infty) - \Phi(-0{,}63) = 0{,}5 - (-0{,}2357) = \mathbf{0{,}7357}$$
|
||||
|
||||
---
|
||||
|
||||
### б) Менее половины (m < 500, то есть m ≤ 499)
|
||||
|
||||
$$P_{1000}(0 \leq m \leq 499) = 1 - 0{,}7357 = \mathbf{0{,}2643}$$
|
||||
|
||||
---
|
||||
|
||||
## Задача 2 (ДЗ — Картофель при уборке)
|
||||
|
||||
**Условие:** При уборке повреждается в среднем 10% клубней. Из **n = 200** клубней найти вероятность того, что повреждено от 15 до 50 клубней.
|
||||
|
||||
Дано: n = 200, p = 0,10, q = 0,90
|
||||
np = 20, npq = 18, √(npq) ≈ 4,243
|
||||
|
||||
$$x_1 = \frac{15 - 20}{4{,}243} \approx -1{,}18, \quad x_2 = \frac{50 - 20}{4{,}243} \approx 7{,}07$$
|
||||
|
||||
$$P_{200}(15 \leq m \leq 50) \approx \Phi(7{,}07) - \Phi(-1{,}18) \approx 0{,}5 - (-0{,}3810) = 0{,}5 + 0{,}3810 \approx \mathbf{0{,}881}$$
|
||||
Reference in New Issue
Block a user