Some checks failed
Build Multi-Platform Binaries / build-binaries (amd64, darwin, server, false) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (amd64, linux, client, true) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (amd64, linux, server, true) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (amd64, windows, client, true) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (amd64, windows, server, true) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (arm, 7, linux, client, true) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (arm, 7, linux, server, true) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (arm64, darwin, server, false) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (arm64, linux, client, true) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (arm64, linux, server, true) (push) Has been cancelled
Build Multi-Platform Binaries / build-binaries (arm64, windows, server, false) (push) Has been cancelled
Build Multi-Platform Binaries / build-frontend (push) Has been cancelled
95 lines
2.1 KiB
Go
95 lines
2.1 KiB
Go
package proxy
|
|
|
|
import (
|
|
"bufio"
|
|
"io"
|
|
"net"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/gotunnel/pkg/relay"
|
|
)
|
|
|
|
// HTTPServer HTTP 代理服务
|
|
type HTTPServer struct {
|
|
dialer Dialer
|
|
onStats func(in, out int64) // 流量统计回调
|
|
}
|
|
|
|
// NewHTTPServer 创建 HTTP 代理服务
|
|
func NewHTTPServer(dialer Dialer, onStats func(in, out int64)) *HTTPServer {
|
|
return &HTTPServer{dialer: dialer, onStats: onStats}
|
|
}
|
|
|
|
// HandleConn 处理 HTTP 代理连接
|
|
func (h *HTTPServer) HandleConn(conn net.Conn) error {
|
|
defer conn.Close()
|
|
|
|
reader := bufio.NewReader(conn)
|
|
req, err := http.ReadRequest(reader)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if req.Method == http.MethodConnect {
|
|
return h.handleConnect(conn, req)
|
|
}
|
|
return h.handleHTTP(conn, req, reader)
|
|
}
|
|
|
|
// handleConnect 处理 CONNECT 方法 (HTTPS)
|
|
func (h *HTTPServer) handleConnect(conn net.Conn, req *http.Request) error {
|
|
target := req.Host
|
|
if !strings.Contains(target, ":") {
|
|
target = target + ":443"
|
|
}
|
|
|
|
remote, err := h.dialer.Dial("tcp", target)
|
|
if err != nil {
|
|
conn.Write([]byte("HTTP/1.1 502 Bad Gateway\r\n\r\n"))
|
|
return err
|
|
}
|
|
defer remote.Close()
|
|
|
|
conn.Write([]byte("HTTP/1.1 200 Connection Established\r\n\r\n"))
|
|
|
|
// 双向转发 (带流量统计)
|
|
relay.RelayWithStats(conn, remote, h.onStats)
|
|
return nil
|
|
}
|
|
|
|
// handleHTTP 处理普通 HTTP 请求
|
|
func (h *HTTPServer) handleHTTP(conn net.Conn, req *http.Request, reader *bufio.Reader) error {
|
|
target := req.Host
|
|
if !strings.Contains(target, ":") {
|
|
target = target + ":80"
|
|
}
|
|
|
|
remote, err := h.dialer.Dial("tcp", target)
|
|
if err != nil {
|
|
conn.Write([]byte("HTTP/1.1 502 Bad Gateway\r\n\r\n"))
|
|
return err
|
|
}
|
|
defer remote.Close()
|
|
|
|
// 修改请求路径为相对路径
|
|
req.URL.Scheme = ""
|
|
req.URL.Host = ""
|
|
req.RequestURI = req.URL.Path
|
|
if req.URL.RawQuery != "" {
|
|
req.RequestURI += "?" + req.URL.RawQuery
|
|
}
|
|
|
|
// 发送请求到目标
|
|
if err := req.Write(remote); err != nil {
|
|
return err
|
|
}
|
|
|
|
// 转发响应 (带流量统计)
|
|
n, err := io.Copy(conn, remote)
|
|
if h.onStats != nil && n > 0 {
|
|
h.onStats(0, n) // 响应数据为出站流量
|
|
}
|
|
return err
|
|
}
|