mirror of
https://github.com/1f349/violet.git
synced 2024-11-23 11:51:37 +00:00
95 lines
2.9 KiB
Go
95 lines
2.9 KiB
Go
package proxy
|
|
|
|
import (
|
|
"crypto/tls"
|
|
"github.com/1f349/violet/proxy/websocket"
|
|
"github.com/google/uuid"
|
|
"log"
|
|
"net"
|
|
"net/http"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
type HybridTransport struct {
|
|
baseDialer *net.Dialer
|
|
normalTransport http.RoundTripper
|
|
insecureTransport http.RoundTripper
|
|
socksSync *sync.RWMutex
|
|
socksTransport map[string]http.RoundTripper
|
|
ws *websocket.Server
|
|
}
|
|
|
|
// NewHybridTransport creates a new hybrid transport
|
|
func NewHybridTransport(ws *websocket.Server) *HybridTransport {
|
|
return NewHybridTransportWithCalls(nil, nil, ws)
|
|
}
|
|
|
|
// NewHybridTransportWithCalls creates new hybrid transport with custom normal
|
|
// and insecure http.RoundTripper functions.
|
|
//
|
|
// NewHybridTransportWithCalls(nil, nil) is equivalent to NewHybridTransport()
|
|
func NewHybridTransportWithCalls(normal, insecure http.RoundTripper, ws *websocket.Server) *HybridTransport {
|
|
h := &HybridTransport{
|
|
baseDialer: &net.Dialer{
|
|
Timeout: 30 * time.Second,
|
|
KeepAlive: 30 * time.Second,
|
|
},
|
|
normalTransport: normal,
|
|
insecureTransport: insecure,
|
|
ws: ws,
|
|
}
|
|
if h.normalTransport == nil {
|
|
h.normalTransport = &http.Transport{
|
|
Proxy: http.ProxyFromEnvironment,
|
|
DialContext: h.baseDialer.DialContext,
|
|
ForceAttemptHTTP2: true,
|
|
MaxIdleConns: 15,
|
|
TLSHandshakeTimeout: 10 * time.Second,
|
|
IdleConnTimeout: 30 * time.Second,
|
|
ExpectContinueTimeout: 1 * time.Second,
|
|
ResponseHeaderTimeout: 10 * time.Second,
|
|
DisableKeepAlives: true,
|
|
}
|
|
}
|
|
if h.insecureTransport == nil {
|
|
h.insecureTransport = &http.Transport{
|
|
Proxy: http.ProxyFromEnvironment,
|
|
DialContext: h.baseDialer.DialContext,
|
|
ForceAttemptHTTP2: true,
|
|
MaxIdleConns: 15,
|
|
TLSHandshakeTimeout: 10 * time.Second,
|
|
IdleConnTimeout: 30 * time.Second,
|
|
ExpectContinueTimeout: 1 * time.Second,
|
|
ResponseHeaderTimeout: 10 * time.Second,
|
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
|
|
DisableKeepAlives: true,
|
|
}
|
|
}
|
|
return h
|
|
}
|
|
|
|
// SecureRoundTrip calls the secure transport
|
|
func (h *HybridTransport) SecureRoundTrip(req *http.Request) (*http.Response, error) {
|
|
u := uuid.New()
|
|
log.Println("[Transport] Start upgrade:", u)
|
|
defer log.Println("[Transport] Stop upgrade:", u)
|
|
return h.normalTransport.RoundTrip(req)
|
|
}
|
|
|
|
// InsecureRoundTrip calls the insecure transport
|
|
func (h *HybridTransport) InsecureRoundTrip(req *http.Request) (*http.Response, error) {
|
|
u := uuid.New()
|
|
log.Println("[Transport insecure] Start upgrade:", u)
|
|
defer log.Println("[Transport insecure] Stop upgrade:", u)
|
|
return h.insecureTransport.RoundTrip(req)
|
|
}
|
|
|
|
// ConnectWebsocket calls the websocket upgrader and thus hijacks the connection
|
|
func (h *HybridTransport) ConnectWebsocket(rw http.ResponseWriter, req *http.Request) {
|
|
u := uuid.New()
|
|
log.Println("[Websocket] Start upgrade:", u)
|
|
h.ws.Upgrade(rw, req)
|
|
log.Println("[Websocket] Stop upgrade:", u)
|
|
}
|