笑死我了、这个都比、应该是见过最长的域名了,架构倒是简单得很 注:本文是站在个人站长的角度,更多的是要实现在已有多个网站的服务器上额外加科学上网服务,而不是在新的服务器上安装 Xray 并顺带一个伪装站点。也就是说,站点为主,Xray 为辅,而不是 Xray 为主【不是先有 Xray,后有站点】。 如果你有一台服务器,既要用来放网站又想用来运行 Xray 服务进行科学上网,应该是很多「国内」草根站长的需求。 6 \! q, O# C+ G- r9 Q+ ~
如果网站方面,仅仅是 nginx 监听 80 端口,也就是仅使用 http 协议,那么两者不会有冲突。但是,如果你想要网站也支持(其实,更多的时候应该是强制)HTTPS,那么就有冲突了,因为 Xray 绑定了 443 端口,nginx 就无法再绑定 443 端口了,反之亦然。 那怎么办呢?肯定不能让任一方使用非 443 端口,这样既不专业也不安全。那么,就只能让其中一个服务把 443 端口全包揽了,无论是科学上网还是站点,都先经由其中一个服务,并由它来分流,然后分流之后的请求再各自到对应的地方(监听端口)。
+ a7 _( w: } b0 G# M
所以,无非是选择让 Xray 还是 nginx 来监听 443 端口,也就是说有两种方案。下面分别举例介绍这两种方案的对应配置。【下面示例中的域名、端口、路径等记得改成自己的】 方案一:nginx 监听 443 端口并分流给后端的 Xray 服务和其他 nginx 站点
这个方案是直觉上的第一方案,因为在需要科学上网之前,nginx 以及其他站点配套服务早就存在了,我不想为了额外加一个 Xray 服务而过多的打乱之前的站点配置。 修改 /etc/nginx/nginx.conf要实现 nginx 监听 443 端口并分流,非常简单,最新版的 nginx 有现成的模块( ngx_stream_ssl_preread_module,👈 nginx 官方文档有详细说明和示例(多种分流方式))。我们需要在 nginx.conf 里添加下面的分流配置代码: 1 2 3 4 5 6 7 8 91011121314151617181920212223242526 | stream { map $ssl_preread_server_name $name { xraydomain.tld xray; # xray 服务使用的域名,这里只能单独列出,不能并排,所以有几个域名就得写上几个,多个站点依次列出即可 www.xraydomain.tld xray; # xray 服务使用的域名 domain1.tld domain1; # 正常 HTTPS 站点 1 www.domain1.tld domain1; # 正常 HTTPS 站点 1 domain2.tld domain2; # 正常 HTTPS 站点 2 www.domain2.tld domain2; # 正常 HTTPS 站点 2 } upstream xray { server 127.0.0.1:1021; # 分流到 Xray 监听端口 } upstream domain1 { server 127.0.0.1:1022; # 分流到站点 1 端口 } upstream domain2 { server 127.0.0.1:1023; # 分流到站点 2 端口 } server { listen 443 reuseport; # listen [::]:443 reuseport; # IPV6,我不需要,所以注释掉了 proxy_pass $name; proxy_protocol on; # 为了获取真实 IP,开启 proxy_protocol,对应的在 default.conf 也需要设置一下 ssl_preread on; }} |
7 k1 C# [5 U" J9 h$ Y
8 W8 ^; [9 d* b& K; _/ Z6 g8 B$ `+ y修改 /etc/nginx/conf.d/default.conf为了获取真实 IP,修改一下 default.conf,顶部添加下面代码: 12 | set_real_ip_from 127.0.0.1;real_ip_header proxy_protocol; | 4 l1 \9 u: y/ A& a3 `; g
8 j7 b# f) D( n* P; l3 ^* pXray 配置文件,/usr/local/etc/xray/config.jsonXray 的配置如下,需要注意两个地方,端口不再写 443 了,改成上面 nginx 分流过来的端口,同样的对应 proxy_protocol,把 acceptProxyProtocol 改成 true。 另外,这里用的是 websocket 方案,其他都类似。 1 2 3 4 5 6 7 8 91011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 | { "log": { "loglevel": "error" }, "inbounds": [ { "port": 1021, "protocol": "vless", "streamSettings": { "network": "tcp", "security": "xtls", "tcpSettings": { "acceptProxyProtocol": true }, "xtlsSettings": { "minVersion": "1.2", "alpn": ["h2", "http/1.1"], "certificates": [ { "certificateFile": "/path/to/xraydomain.crt", "keyFile": "/path/to/xraydomain.key" } ] } }, "settings": { "clients": [ { "id": "U-U-I-D", "flow": "xtls-rprx-direct", "level": 0, } ], "decryption": "none", "fallbacks": [ { "name": "xraydomain.tld", "dest": 10211, "xver": 1 }, { "name": "xraydomain.tld", "alpn": "h2", "dest": 10212, "xver": 1 }, { "name": "www.xraydomain.tld", "dest": 10211, "xver": 1 }, { "name": "www.xraydomain.tld", "alpn": "h2", "dest": 10212, "xver": 1 }, { "path": "/vlessws", "dest": 1234, "xver": 1 } ] } }, { "port": 1234, "listen": "127.0.0.1", "protocol": "vless", "settings": { "clients": [ { "id": "U-U-I-D", "level": 0, } ], "decryption": "none" }, "streamSettings": { "network": "ws", "security": "none", "wsSettings": { "acceptProxyProtocol": true, "path": "/vlessws" } } } ], "outbounds": [ { "protocol": "freedom" } ]} | $ ]5 k4 {- p% d
% e7 M& s2 {+ J) B7 [( i R
nginx 正常 HTTPS 站点的配置,例如站点 1,/etc/nginx/sites-available/domain1.tld需要说明一点,看第 8 和 16 行,按理说这里应该是 listen 127.0.0.1:1022 ssl http2 proxy_protocol;,与上面 nginx.conf 对应,可是我在自己的 VPS 上测试,这样写会出错,可能与我的系统环境有关,我的环境中 nginx 实际监听的是 0.0.0.0:443,分流过来的也是 0.0.0.0:1022,如果改成 127.0.0.1 会出错,所以我直接写成了 1022。 1 2 3 4 5 6 7 8 910111213141516171819202122 | server { listen 80; server_name domain1.tld www.domain1.tld; return 301 https://domain1.tld$request_uri;}server { listen 1022 ssl http2 proxy_protocol; server_name www.domain1.tld; ssl_certificate /path/to/domain1.crt; ssl_certificate_key /path/to/domain1.key; return 301 https://domain1.tld$request_uri;}server { listen 1022 ssl http2 proxy_protocol; server_name domain1.tld; ssl_certificate /path/to/domain1.crt; ssl_certificate_key /path/to/domain1.key; # 注意,下面还有更多的 sll 相关配置以及站点其他配置, # 这里省略了,自己站点之前怎么配置的,接着用就行,无需改动。} | + I; j$ n: m5 s
9 ^6 K$ S1 x+ i# e h2 n
nginx Xray 伪装站点的配置,/etc/nginx/sites-available/xraydomain.tld 1 2 3 4 5 6 7 8 9101112 | server { listen 80; server_name xraydomain.tld www.xraydomain.tld; return 301 https://xraydomain.tld$request_uri;}server { listen 10211 proxy_protocol; listen 10212 proxy_protocol http2; server_name xraydomain.tld www.xraydomain.tld; # 注意,下面还有更多的站点配置,root、error_log 等等这里省略了,请根据自己的情况配置。} |
* @! |. v q4 _0 B1 N' N: C+ T, x# A7 C% F: y9 n/ g; q
方案二:Xray 监听 443 端口并分流给后端的 nginx 站点(包括伪装站)这个方案是让 Xray 在前端,负责监听 443 端口,并使用 SNI 机制配合 fallbacks 规则实现分流,nginx 站点(包括伪装站)在后面接应。所以,如果有很多个站点的话,都得罗列到 Xray 的 config.json 里,包括域名证书、fallbacks 等,最终这个文件会很长,注意不要出错。 主要是不确定 fallbacks 里面的域名能不能并列着写,如果可以的话,还能减少一些行数,否则就会像下面这样,站点越多越长。 另外,这里用的是 websocket 方案,其他都类似。 Xray 负责监听 443 端口并分流,/usr/local/etc/xray/config.json 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147 | { "log": { "loglevel": "error" }, "inbounds": [ { "port": 443, "protocol": "vless", "streamSettings": { "network": "tcp", "security": "xtls", "xtlsSettings": { "minVersion": "1.2", "alpn": ["h2", "http/1.1"], "certificates": [ { "certificateFile": "/path/to/xraydomain.crt", "keyFile": "/path/to/xraydomain.key" }, { "certificateFile": "/path/to/domain1.crt", "keyFile": "/path/to/domain1.key" }, { "certificateFile": "/path/to/domain2.crt", "keyFile": "/path/to/domain2.key" } ] } }, "settings": { "clients": [ { "id": "U-U-I-D", "flow": "xtls-rprx-direct", "level": 0, } ], "decryption": "none", "fallbacks": [ { "name": "xraydomain.tld", "dest": 10211, "xver": 1 }, { "name": "xraydomain.tld", "alpn": "h2", "dest": 10212, "xver": 1 }, { "name": "www.xraydomain.tld", "dest": 10211, "xver": 1 }, { "name": "www.xraydomain.tld", "alpn": "h2", "dest": 10212, "xver": 1 }, { "path": "/vlessws", "dest": 1234, "xver": 1 }, { "name": "domain1.tld", "dest": 10221, "xver": 1 }, { "name": "domain1.tld", "alpn": "h2", "dest": 10222, "xver": 1 }, { "name": "www.domain1.tld", "dest": 10221, "xver": 1 }, { "name": "www.domain1.tld", "alpn": "h2", "dest": 10222, "xver": 1 }, { "name": "domain2.tld", "dest": 10231, "xver": 1 }, { "name": "domain2.tld", "alpn": "h2", "dest": 10232, "xver": 1 }, { "name": "www.domain2.tld", "dest": 10231, "xver": 1 }, { "name": "www.domain2.tld", "alpn": "h2", "dest": 10232, "xver": 1 } ] } }, { "port": 1234, "listen": "127.0.0.1", "protocol": "vless", "settings": { "clients": [ { "id": "U-U-I-D", "level": 0, } ], "decryption": "none" }, "streamSettings": { "network": "ws", "security": "none", "wsSettings": { "acceptProxyProtocol": true, "path": "/vlessws" } } } ], "outbounds": [ { "protocol": "freedom" } ]} |
2 G+ ?* y" T( o( T, o0 W$ c5 W) |" R
修改 nginx default 配置,/etc/nginx/conf.d/default.conf为了获取真实 IP,修改一下 default.conf,顶部添加下面代码: 12 | set_real_ip_from 127.0.0.1;real_ip_header proxy_protocol; |
" x( J/ ]+ `4 z& y4 z2 G6 X3 ^+ \$ O4 E5 W
Xray 伪装站点的配置,/etc/nginx/sites-available/xraydomain.tld 1 2 3 4 5 6 7 8 9101112 | server { listen 80; server_name xraydomain.tld www.xraydomain.tld; return 301 https://xraydomain.tld$request_uri;}server { listen 127.0.0.1:10211 proxy_protocol; listen 127.0.0.1:10212 proxy_protocol http2; server_name xraydomain.tld www.xraydomain.tld; # 注意,下面还有更多的站点配置,root、error_log 等等这里省略了,请根据自己的情况配置。} | 2 X7 H0 A6 r( X; i
+ I5 Z$ `6 d. U% ^0 wnginx 正常 HTTPS 站点的配置,例如站点 2,/etc/nginx/sites-available/domain2.tld需要注意的是,nginx 需要用两个端口分别来监听 HTTP/1.1 和 HTTP/2,前面的 Xray 分流已经给区分开了,带着 "alpn": "h2", 的即是 HTTP/2协议。 1 2 3 4 5 6 7 8 9101112131415161718192021222324 | |
& I1 i0 b; h+ ^+ a# V: c4 yXray与多个HTTPS站点(Nginx)共存,两种443端口复用方案 - 离离原上草 (liliyuanshangcaoyisuiyikurongyehuoshaobujinchunfengchuiyousheng.com) |