Awesome
基于 hyper
和 rustls
的静态资源托管服务器、正向代理、反向代理。
功能特性
- 使用tls来对正向代理流量进行加密(
--over-tls
)。 - 类Nginx的静态资源托管。支持gzip压缩。支持Accept-Ranges以支持断点续传(备注:暂不支持多range,例如
Range: bytes=0-100,100-
) - 支持反向代理(
--reverse-proxy-config-file
)。 - 基于Prometheus的可观测,可以监控代理的流量、外链访问等。
- 采集网卡上行流量,展示在
/speed
路径下(读取/proc/net/dev
或基于ebpf socket filter
) - 支持多端口,多用户。
- 每天定时加载tls证书,acme证书过期重新签发时不需要重启服务。
- 连接空闲(10分钟没有IO)自动关闭。
提及的参数详见命令行参数
安装说明
linux amd64 可执行文件
curl -SLf https://us.arloor.dev/https://github.com/arloor/rust_http_proxy/releases/download/latest/rust_http_proxy -o /tmp/rust_http_proxy
install /tmp/rust_http_proxy /usr/bin/rust_http_proxy
/usr/bin/rust_http_proxy -p 7788
Docker 安装
通过Github Action自动更新release,永远是最新版,可放心使用
docker run --rm -it --net host docker.io/arloor/rust_http_proxy -p 7788
ebpf版本安装
注意:ebpf版本并没有什么新特性,只是为了学习下新技术的自嗨。
curl -SLf https://us.arloor.dev/https://github.com/arloor/rust_http_proxy/releases/download/bpf/rust_http_proxy -o /tmp/rust_http_proxy
install /tmp/rust_http_proxy /usr/bin/rust_http_proxy
/usr/bin/rust_http_proxy -p 7788
或者
docker run --rm -it --privileged --net host docker.io/arloor/rust_http_proxy:bpf -p 7788
命令行参数
$ rust_http_proxy --help
A HTTP proxy server based on Hyper and Rustls, which features TLS proxy and static file serving
Usage: rust_http_proxy [OPTIONS]
Options:
--log-dir <LOG_DIR>
[default: /tmp]
--log-file <LOG_FILE>
[default: proxy.log]
-p, --port <PORT>
可以多次指定来实现多端口
[default: 3128]
-c, --cert <CERT>
[default: cert.pem]
-k, --key <KEY>
[default: privkey.pem]
-u, --users <USER>
默认为空,表示不鉴权。
格式为 'username:password'
可以多次指定来实现多用户
-w, --web-content-path <WEB_CONTENT_PATH>
[default: /usr/share/nginx/html]
-r, --referer-keywords-to-self <REFERER>
Http Referer请求头处理
1. 图片资源的防盗链:针对png/jpeg/jpg等文件的请求,要求Request的Referer header要么为空,要么包含配置的值
2. 外链访问监控:如果Referer不包含配置的值,并且访问html资源时,Prometheus counter req_from_out++,用于外链访问监控
可以多次指定,也可以不指定
--never-ask-for-auth
if enable, never send '407 Proxy Authentication Required' to client。
建议设置为true,否则有被嗅探的风险
-o, --over-tls
if enable, proxy server will listen on https
--hostname <HOSTNAME>
[default: unknown]
--reverse-proxy-config-file <FILE_PATH>
反向代理配置文件
--enable-github-proxy
是否开启github proxy
--append-upstream-url <https://example.com>
便捷反向代理配置
例如:--append-upstream-url=https://cdnjs.cloudflare.com
则访问 https://your_domain/https://cdnjs.cloudflare.com 会被代理到 https://cdnjs.cloudflare.com
-h, --help
Print help
SSL配置
其中,tls证书(--cert
)和pem格式的私钥(--key
)可以通过openssl命令一键生成:
openssl req -x509 -newkey rsa:4096 -sha256 -nodes -keyout /usr/share/rust_http_proxy/privkey.pem -out /usr/share/rust_http_proxy/cert.pem -days 3650 -subj "/C=cn/ST=hl/L=sd/O=op/OU=as/CN=example.com"
如需签名证书,请购买tls证书或免费解决方案(acme.sh等)
测试TLS Proxy可以使用curl (7.52.0以上版本):
curl https://ip.im/info -U "username:password" -x https://localhost:7788 --proxy-insecure
反向代理配置
YOUR_DOMAIN:
- location: / # 默认为 /
upstream:
url_base: https://www.baidu.com
version: H1 # 可以填H1、H2、AUTO,默认为AUTO
如果
YOUR_DOMAIN
填default_host
则对所有的域名生效
例子1: Github Proxy
在github原始url前加上https://YOUR_DOMAIN
,以便在国内访问raw.githubusercontent.com、github.com和gist.githubusercontent.com
启动参数中增加 --enable-github-proxy
,相当于以下配置:
default_host:
- location: /https://gist.githubusercontent.com
upstream:
url_base: https://gist.githubusercontent.com
- location: /https://gist.github.com
upstream:
url_base: https://gist.github.com
- location: /https://github.com
upstream:
url_base: https://github.com
- location: /https://objects.githubusercontent.com
upstream:
url_base: https://objects.githubusercontent.com
- location: /https://raw.githubusercontent.com
upstream:
url_base: https://raw.githubusercontent.com
例子2: 反向代理https://cdnjs.cloudflare.com
启动参数中增加 --append-upstream-url=https://cdnjs.cloudflare.com
,相当于以下配置:
default_host:
- location: /https://cdnjs.cloudflare.com
upstream:
url_base: https://cdnjs.cloudflare.com
例子3: 改写Github Models的url为openai api的url格式
default_host:
- location: /v1/chat/completions
upstream:
url_base: https://models.inference.ai.azure.com/chat/completions
可观测
Prometheus Exporter
提供了Prometheus的Exporter。如果设置了--users
参数,则需要在header中设置authorization,否则会返回401 UNAUTHORIZED
。
# HELP req_from_out Number of HTTP requests received.
# TYPE req_from_out counter
req_from_out_total{referer="all",path="all"} 4
# HELP proxy_traffic num proxy_traffic.
# TYPE proxy_traffic counter
# EOF
可以使用此Grafana大盘Template来创建Grafana大盘,效果如下
Linux运行时的网速监控
在linux运行时,会监控网卡网速,并展示在 /net
。
客户端
- Clash系列
- 自研玩具
- Rust:sslocal(fork shadowsocks-rust)
- Golang:forward
- Java: connect
Cargo Features
bpf
使用ebpf来统计网卡出流量,仅在 x86_64-unknown-linux-gnu
上测试过。激活方式:
cargo build --features bpf
需要安装 libbpf-rs
所需的依赖:
ubuntu 22.04 安装:
apt-get install -y libbpf-dev bpftool cmake zlib1g-dev libelf-dev pkg-config clang autoconf autopoint flex bison gawk make
centos stream 9 安装:
yum install -y libbpf zlib-devel elfutils-libelf-devel pkgconf-pkg-config clang bpftool cmake autoconf gettext flex bison gawk make
jemalloc
拥有更高的并发分配能力和减少内存碎片,不过会buffer更多的内存,因此top中RES数值会有上升。激活方式:
cargo build --features jemalloc
aws_lc_rs
aws_lc_rs
和 ring
是 rustls
的两个加密后端。本项目默认使用 ring
作为加密后端,也可选择aws_lc_rs作为加密后端。aws_lc_rs
相比ring主要有两点优势:
- 在rustls的benchmark测试中,
aws_lc_rs
的性能要优于ring
。 - 支持美国联邦政府针对加密提出的fips要求。
不过,使用 aws_lc_rs
会增加一些编译难度,需要额外做以下操作:
依赖的包 | 是否必须 | 安装方式 |
---|---|---|
cmake | 必须 | apt-get install cmake |
激活方式:
cargo build --no-default-features --features aws_lc_rs
高匿实现
代理服务器收到的http请求有一些特征,如果代理服务器不能正确处理,则会暴露自己是一个代理。高匿代理就是能去除这些特征的代理。具体特征有三个:
- 代理服务器收到的request line中有完整url,即包含schema、host。而正常http请求的url只包含路径
- 代理服务器收到http header中有Proxy-Connection请求头,需要去掉
- 代理服务器收到http header中有Proxy-Authentication请求头,需要去掉
本代理能去除以上特征。下面是使用tcpdump测试的结果,分别展示代理服务器收到的http请求和nginx web服务器收到的http请求已验证去除以上特征。
代理服务器收到的消息:
Nginx收到的消息:
可以看到请求URL和Proxy-Connection
都被正确处理了。