aixiao 767ada5e43 feat(blacklist): 支持国际化域名与 hosts 文件风格黑名单
- 引入 `golang.org/x/net/idna` 实现 Unicode 域名转 ASCII(Punycode)
- 黑名单加载支持通配符格式如 `*.example.com`
- 支持解析 hosts 风格的文件(每行首列为 IP 地址时,其余列为域名)
- 扩展 Scanner 缓冲区至 2MB 以适应大型 hosts 文件
- 注释处理优化,兼容 `#` 和 `;` 分隔符
- 加载后对规则排序并去重,提升匹配效率与一致性

fix(cache): 调整负面响应缓存逻辑与上游查询并发控制

- 明确区分 NXDOMAIN 与 NODATA 并正确处理 SOA 缺失情况
- 查询上游时引入更可靠的并发限制与超时机制
- UDP 截断时自动回退 TCP 查询
- 过滤无效 RCODE(如 SERVFAIL、REFUSED 等),防止污染结果
- 区分“全部失败”与“部分完成但无有效响应”,增强调试日志信息
2025-10-17 17:28:44 +08:00

DNS-over-TLS Cache Proxy

一个用 Go 编写的高性能 DNS-over-TLS (DoT) 缓存代理服务,
专为隐私保护与性能优化而设计。支持多上游并发解析、智能缓存、ECS 剥离和优雅关闭。

🚀 特性概览

功能 描述
🔒 加密传输 完全支持 DNS-over-TLSRFC 7858支持 TLS 1.2/1.3
多上游并发解析 “快乐眼球”机制并行查询多个上游 DNS取最快响应
🧠 智能缓存系统 支持正向与负面缓存RFC 2308动态 TTL 调整
🧹 自动清理机制 定期清理过期缓存项(默认每 5 分钟)
🧩 隐私保护 默认剥离 ECS (EDNS Client Subnet),防止地理泄露
🧱 黑名单过滤 支持域名黑名单(后缀匹配、文件或命令行加载)
🪶 轻量高效 单一可执行文件,无外部依赖,易于容器化部署

🏗️ 快速开始

从源码构建

git clone https://git.aixiao.me/aixiao/dot.git
cd dot
bash build.sh bin

使用 Docker 构建与运行

# 构建镜像
bash build.sh build

# 运行容器
bash build.sh run

# 查看日志
bash build.sh logs

# 停止与清理
bash build.sh stop
bash build.sh clean

🧩 默认镜像名为 dot:latest,监听 853 端口,可通过 PORT 环境变量修改。

⚙️ 启动示例

./dot \
    -cert aixiao.me.cer \
    -key aixiao.me.key \
    -addr :853 \
    -upstream 119.29.29.29:53,223.5.5.5:53,114.114.114.114:53 \
    -cache-ttl 300s \
    -timeout 3s \
    -max-parallel 3 \
    -blacklist-file blacklist.txt

启动日志示例:

🚀 starting DNS-over-TLS on :853
[req] A www.example.com. (id=40192 cd=false do=true from=127.0.0.1:58877)
[cache] MISS A www.example.com.
[answer] www.example.com. 300 IN A 93.184.216.34

⚙️ 参数说明

参数 默认值 描述
--addr :853 监听地址(支持 IPv4/IPv6
--cert server.crt TLS 证书路径
--key server.key TLS 私钥路径
--upstream 8.8.8.8:53,1.1.1.1:53 上游 DNS 服务器列表
--cache-ttl 60s 缓存最大 TTL正向/负面均适用)
--timeout 3s 上游查询超时
--max-parallel 3 最大并发上游查询数
--strip-ecs true 是否剥离 ECS 信息
--tcp-fallback true UDP 截断时是否自动 TCP 回退
--blacklist 逗号分隔的黑名单域名或通配后缀
--blacklist-file 黑名单文件路径(每行一个规则)
--blacklist-rcode REFUSED 黑名单命中返回码:REFUSED / NXDOMAIN / SERVFAIL
--cache-size 10000 LRU 缓存最大条目数
--v false 启用详细日志模式

🔍 缓存机制详解

缓存键格式:

domain|type|class|DO|CD

缓存策略:

  • 正向缓存:取最小 TTL 与配置上限的较小值
  • 🚫 负面缓存:遵循 RFC 2308从 SOA.MINIMUM 计算 TTL
  • 🧭 动态 TTL 调整:返回时按剩余时间递减 TTL
  • 🧹 自动清理:每 5 分钟扫描并删除过期条目
  • 🔒 隔离逻辑DO/CD 不同查询独立缓存空间

🧱 黑名单功能

支持两种配置方式:

  1. 命令行参数:

    ./dot -blacklist="*.ads.com,*.tracking.net"
    
  2. 文件加载(每行一个域名或后缀):

    # blacklist.txt
    *.ads.com
    *.malware.net
    

    启动命令:

    ./dot -blacklist-file=blacklist.txt -blacklist-rcode=NXDOMAIN
    

黑名单命中后不再上游查询,直接返回指定 RCODE。

🧩 架构与运行原理

flowchart TD
  A[Client (DoT Request)] --> B[DNS-over-TLS Server]
  B --> C[Cache Lookup]
  C -- HIT --> D[Return Cached Response]
  C -- MISS --> E[Upstream Resolver Pool]
  E -->|Fastest Response| F[DNS Response]
  F --> G[Cache Write]
  G --> H[Return to Client]

🧰 开发与维护

🧪 测试方法

使用 kdigdig 测试解析:

kdig @127.0.0.1 +tls-ca +tls-host=dot.local www.example.com

👨‍💻 作者信息

Author: niuyuling
Email: aixiao@aixiao.me
License: MIT
Repository: git.aixiao.me/aixiao/dot

Description
No description provided
Readme 43 MiB
Languages
Go 87.4%
Shell 9.4%
Dockerfile 3.2%