Files
openlan/pkg/proxy/http.go
2024-01-24 20:16:50 +08:00

308 lines
6.9 KiB
Go
Executable File

package proxy
import (
"bufio"
"bytes"
"crypto/tls"
"encoding/base64"
"fmt"
"io"
"net"
"net/http"
"net/http/httputil"
"strings"
"time"
"github.com/luscis/openlan/pkg/config"
"github.com/luscis/openlan/pkg/libol"
)
type HttpProxy struct {
users map[string]string
out *libol.SubLogger
server *http.Server
cfg *config.HttpProxy
}
var (
connectOkay = []byte("HTTP/1.1 200 Connection established\r\n\r\n")
)
func parseBasicAuth(auth string) (username, password string, ok bool) {
const prefix = "Basic "
if len(auth) < len(prefix) || !strings.EqualFold(auth[:len(prefix)], prefix) {
return
}
c, err := base64.StdEncoding.DecodeString(auth[len(prefix):])
if err != nil {
return
}
cs := string(c)
s := strings.IndexByte(cs, ':')
if s < 0 {
return
}
return cs[:s], cs[s+1:], true
}
func NewHttpProxy(cfg *config.HttpProxy) *HttpProxy {
h := &HttpProxy{
out: libol.NewSubLogger(cfg.Listen),
cfg: cfg,
users: make(map[string]string),
}
h.server = &http.Server{
Addr: cfg.Listen,
Handler: h,
}
auth := cfg.Auth
if auth.Username != "" {
h.users[auth.Username] = auth.Password
}
h.LoadPass()
return h
}
func (t *HttpProxy) LoadPass() {
file := t.cfg.Password
if file == "" {
return
}
reader, err := libol.OpenRead(file)
if err != nil {
libol.Warn("HttpProxy.LoadPass open %v", err)
return
}
defer reader.Close()
scanner := bufio.NewScanner(reader)
for scanner.Scan() {
line := scanner.Text()
columns := strings.SplitN(line, ":", 2)
if len(columns) < 2 {
continue
}
user := columns[0]
pass := columns[1]
t.users[user] = pass
}
if err := scanner.Err(); err != nil {
libol.Warn("HttpProxy.LoadPass scaner %v", err)
}
}
func (t *HttpProxy) isAuth(username, password string) bool {
if p, ok := t.users[username]; ok {
return p == password
}
return false
}
func (t *HttpProxy) CheckAuth(w http.ResponseWriter, r *http.Request) bool {
if len(t.users) == 0 {
return true
}
auth := r.Header.Get("Proxy-Authorization")
user, password, ok := parseBasicAuth(auth)
if !ok || !t.isAuth(user, password) {
w.Header().Set("Proxy-Authenticate", "Basic")
http.Error(w, "Proxy Authentication Required", http.StatusProxyAuthRequired)
return false
}
return true
}
func (t *HttpProxy) route(w http.ResponseWriter, p *http.Response) {
defer p.Body.Close()
for key, value := range p.Header {
if key == "Proxy-Authorization" {
if len(value) > 0 { // Remove first value for next proxy.
value = value[1:]
}
}
for _, v := range value {
w.Header().Add(key, v)
}
}
w.WriteHeader(p.StatusCode)
_, _ = io.Copy(w, p.Body)
}
func (t *HttpProxy) tunnel(w http.ResponseWriter, conn net.Conn) {
src, bio, err := w.(http.Hijacker).Hijack()
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
defer src.Close()
wait := libol.NewWaitOne(2)
libol.Go(func() {
defer wait.Done()
// The returned bufio.Reader may contain unprocessed buffered data from the client.
// Copy them to dst so we can use src directly.
if n := bio.Reader.Buffered(); n > 0 {
n64, err := io.CopyN(conn, bio, int64(n))
if n64 != int64(n) || err != nil {
t.out.Warn("HttpProxy.tunnel io.CopyN:", n64, err)
return
}
}
if _, err := io.Copy(conn, src); err != nil {
t.out.Debug("HttpProxy.tunnel from ws %s", err)
}
})
libol.Go(func() {
defer wait.Done()
if _, err := io.Copy(src, conn); err != nil {
t.out.Debug("HttpProxy.tunnel from target %s", err)
}
})
wait.Wait()
t.out.Debug("HttpProxy.tunnel %s exit", conn.RemoteAddr())
}
func (t *HttpProxy) openConn(remote string) (net.Conn, error) {
if strings.HasPrefix(remote, "https://") {
conf := &tls.Config{
InsecureSkipVerify: true,
}
return tls.Dial("tcp", remote[8:], conf)
} else if strings.HasPrefix(remote, "http://") {
remote = remote[7:]
}
return net.Dial("tcp", remote)
}
func (t *HttpProxy) dumpRequest(r *http.Request) ([]byte, error) {
var err error
var b bytes.Buffer
reqURI := r.RequestURI
if reqURI == "" {
reqURI = r.URL.RequestURI()
}
fmt.Fprintf(&b, "%s %s HTTP/%d.%d\r\n", r.Method, reqURI, r.ProtoMajor, r.ProtoMinor)
host := r.Host
if host == "" && r.URL != nil {
host = r.URL.Host
}
if host != "" {
fmt.Fprintf(&b, "Host: %s\r\n", host)
}
chunked := len(r.TransferEncoding) > 0 && r.TransferEncoding[0] == "chunked"
if len(r.TransferEncoding) > 0 {
fmt.Fprintf(&b, "Transfer-Encoding: %s\r\n", strings.Join(r.TransferEncoding, ","))
}
if r.Close {
fmt.Fprintf(&b, "Connection: close\r\n")
}
err = r.Header.WriteSubset(&b, nil)
if err != nil {
return nil, err
}
io.WriteString(&b, "\r\n")
if r.Body != nil {
var dest io.Writer = &b
if chunked {
dest = httputil.NewChunkedWriter(dest)
}
_, err = io.Copy(dest, r.Body)
if chunked {
dest.(io.Closer).Close()
io.WriteString(&b, "\r\n")
}
}
if err != nil {
return nil, err
}
return b.Bytes(), nil
}
func (t *HttpProxy) ServeHTTP(w http.ResponseWriter, r *http.Request) {
t.out.Debug("HttpProxy.ServeHTTP %v", r)
t.out.Debug("HttpProxy.ServeHTTP %v", r.URL.Host)
if !t.CheckAuth(w, r) {
t.out.Info("HttpProxy.ServeHTTP Required %v Authentication", r.URL.Host)
return
}
forward := t.cfg.Forward
if forward != "" {
t.out.Info("HttpProxy.ServeHTTP %s %s -> %s via %s", r.Method, r.RemoteAddr, r.URL.Host, forward)
} else {
t.out.Info("HttpProxy.ServeHTTP %s %s -> %s", r.Method, r.RemoteAddr, r.URL.Host)
}
if forward != "" {
conn, err := t.openConn(forward)
if err != nil {
http.Error(w, err.Error(), http.StatusBadGateway)
return
}
dump, err := t.dumpRequest(r)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
conn.Write(dump)
t.tunnel(w, conn)
} else if r.Method == "CONNECT" { //RFC-7231 Tunneling TCP based protocols through Web Proxy servers
conn, err := t.openConn(r.URL.Host)
if err != nil {
http.Error(w, err.Error(), http.StatusBadGateway)
return
}
w.Write(connectOkay)
t.tunnel(w, conn)
} else { //RFC 7230 - HTTP/1.1: Message Syntax and Routing
transport := &http.Transport{}
p, err := transport.RoundTrip(r)
if err != nil {
http.Error(w, err.Error(), http.StatusBadGateway)
return
}
defer transport.CloseIdleConnections()
t.route(w, p)
}
}
func (t *HttpProxy) Start() {
if t.server == nil || t.cfg == nil {
return
}
crt := t.cfg.Cert
if crt == nil || crt.KeyFile == "" {
t.out.Info("HttpProxy.start http://%s", t.server.Addr)
} else {
t.out.Info("HttpProxy.start https://%s", t.server.Addr)
}
promise := &libol.Promise{
First: time.Second * 2,
MaxInt: time.Minute,
MinInt: time.Second * 10,
}
promise.Go(func() error {
if crt == nil || crt.KeyFile == "" {
if err := t.server.ListenAndServe(); err != nil {
t.out.Warn("HttpProxy.start %s", err)
return err
}
} else {
if err := t.server.ListenAndServeTLS(crt.CrtFile, crt.KeyFile); err != nil {
t.out.Error("HttpProxy.start %s", err)
return err
}
}
t.server.Shutdown(nil)
return nil
})
}