mirror of
https://gitclone.com/github.com/MetaCubeX/Clash.Meta
synced 2024-11-15 13:41:23 +08:00
78 lines
1.4 KiB
Go
78 lines
1.4 KiB
Go
package http
|
|
|
|
import (
|
|
"io"
|
|
"net"
|
|
"net/http"
|
|
"time"
|
|
|
|
C "github.com/Dreamacro/clash/constant"
|
|
)
|
|
|
|
type HttpAdapter struct {
|
|
addr *C.Addr
|
|
r *http.Request
|
|
w http.ResponseWriter
|
|
done chan struct{}
|
|
}
|
|
|
|
func (h *HttpAdapter) Close() {
|
|
h.done <- struct{}{}
|
|
}
|
|
|
|
func (h *HttpAdapter) Addr() *C.Addr {
|
|
return h.addr
|
|
}
|
|
|
|
func (h *HttpAdapter) Connect(proxy C.ProxyAdapter) {
|
|
req := http.Transport{
|
|
Dial: func(string, string) (net.Conn, error) {
|
|
return proxy.Conn(), nil
|
|
},
|
|
// from http.DefaultTransport
|
|
MaxIdleConns: 100,
|
|
IdleConnTimeout: 90 * time.Second,
|
|
ExpectContinueTimeout: 1 * time.Second,
|
|
}
|
|
resp, err := req.RoundTrip(h.r)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
header := h.w.Header()
|
|
for k, vv := range resp.Header {
|
|
for _, v := range vv {
|
|
header.Add(k, v)
|
|
}
|
|
}
|
|
h.w.WriteHeader(resp.StatusCode)
|
|
var writer io.Writer = h.w
|
|
if len(resp.TransferEncoding) > 0 && resp.TransferEncoding[0] == "chunked" {
|
|
writer = ChunkWriter{Writer: h.w}
|
|
}
|
|
io.Copy(writer, resp.Body)
|
|
}
|
|
|
|
type ChunkWriter struct {
|
|
io.Writer
|
|
}
|
|
|
|
func (cw ChunkWriter) Write(b []byte) (int, error) {
|
|
n, err := cw.Writer.Write(b)
|
|
if err == nil {
|
|
cw.Writer.(http.Flusher).Flush()
|
|
}
|
|
return n, err
|
|
}
|
|
|
|
func NewHttp(host string, w http.ResponseWriter, r *http.Request) (*HttpAdapter, chan struct{}) {
|
|
done := make(chan struct{})
|
|
return &HttpAdapter{
|
|
addr: parseHttpAddr(host),
|
|
r: r,
|
|
w: w,
|
|
done: done,
|
|
}, done
|
|
}
|