split file

master
dingjun 4 years ago
parent 78a61c187e
commit 9a7e25c408

@ -3,7 +3,6 @@ package main
import ( import (
"flag" "flag"
"io/ioutil" "io/ioutil"
"net/url"
"os" "os"
"os/signal" "os/signal"
"syscall" "syscall"
@ -12,45 +11,6 @@ import (
"gopkg.in/yaml.v2" "gopkg.in/yaml.v2"
) )
func makeServers(cfg conf) {
var wsservers = []wsServer{}
var tcpservers = []tcpServer{}
for _, c := range cfg.ProxyConfig {
u, err := url.Parse(c.Listen)
if err != nil {
log.Fatalf("parse %s, error %s", c.Listen, err)
}
switch u.Scheme {
case "ws":
exists := false
for i := 0; i < len(wsservers); i++ {
if wsservers[i].addr == u.Host {
exists = true
wsservers[i].rule = append(wsservers[i].rule, forwardRule{u.Path, c.Remote})
break
}
}
if !exists {
wsservers = append(wsservers, wsServer{u.Host, []forwardRule{{u.Path, c.Remote}}})
}
case "tcp":
tcpservers = append(tcpservers, tcpServer{u.Host, c.Remote})
default:
log.Fatalf("unsupported scheme %s", u.Scheme)
}
}
for _, srv := range wsservers {
go srv.run()
}
for _, srv := range tcpservers {
go srv.run()
}
}
func main() { func main() {
var cfgfile string var cfgfile string
var logfile string var logfile string

@ -3,52 +3,23 @@ package main
import ( import (
"io" "io"
"net" "net"
"net/http"
"net/url" "net/url"
log "github.com/fangdingjun/go-log/v5" log "github.com/fangdingjun/go-log/v5"
"github.com/gorilla/websocket" "github.com/gorilla/websocket"
) )
type forwardRule struct { func forwardWS2WS(conn1, conn2 *websocket.Conn) {
local string
remote string
}
type wsServer struct {
addr string
rule []forwardRule
}
type tcpServer struct {
addr string
remote string
}
func (wss *wsServer) run() {
if err := http.ListenAndServe(wss.addr, wss); err != nil {
log.Errorln(err)
}
}
var upgrader = websocket.Upgrader{
ReadBufferSize: 1024,
WriteBufferSize: 1024,
}
var dialer = &websocket.Dialer{}
func forwardWS2WS(conn, conn1 *websocket.Conn) {
ch := make(chan struct{}, 2) ch := make(chan struct{}, 2)
go func() { go func() {
for { for {
t, data, err := conn.ReadMessage() t, data, err := conn1.ReadMessage()
if err != nil { if err != nil {
log.Errorln(err) log.Errorln(err)
break break
} }
err = conn1.WriteMessage(t, data) err = conn2.WriteMessage(t, data)
if err != nil { if err != nil {
log.Errorln(err) log.Errorln(err)
break break
@ -59,12 +30,12 @@ func forwardWS2WS(conn, conn1 *websocket.Conn) {
go func() { go func() {
for { for {
t, data, err := conn1.ReadMessage() t, data, err := conn2.ReadMessage()
if err != nil { if err != nil {
log.Errorln(err) log.Errorln(err)
break break
} }
err = conn.WriteMessage(t, data) err = conn1.WriteMessage(t, data)
if err != nil { if err != nil {
log.Errorln(err) log.Errorln(err)
break break
@ -140,112 +111,41 @@ func forwardTCP2TCP(c1, c2 net.Conn) {
<-ch <-ch
} }
func (wss *wsServer) ServeHTTP(w http.ResponseWriter, r *http.Request) { func makeServers(cfg conf) {
p := r.URL.Path var wsservers = []wsServer{}
remote := "" var tcpservers = []tcpServer{}
for _, ru := range wss.rule {
if ru.local == p {
remote = ru.remote
}
}
if remote == "" {
http.Error(w, "not found", http.StatusNotFound)
return
}
conn, err := upgrader.Upgrade(w, r, nil)
if err != nil {
log.Errorln(err)
http.Error(w, "bad request", http.StatusBadRequest)
return
}
defer conn.Close()
u, _ := url.Parse(remote)
if u.Scheme == "ws" || u.Scheme == "wss" {
conn1, resp, err := dialer.Dial(remote, nil)
if err != nil {
log.Errorln(err)
return
}
resp.Body.Close()
if resp.StatusCode != http.StatusSwitchingProtocols {
log.Errorf("dial remote ws %d", resp.StatusCode)
return
}
defer conn1.Close()
forwardWS2WS(conn, conn1)
return
}
if u.Scheme == "tcp" {
conn1, err := net.Dial("tcp", u.Host)
if err != nil {
log.Errorln(err)
return
}
defer conn1.Close()
forwardWS2TCP(conn, conn1) for _, c := range cfg.ProxyConfig {
return u, err := url.Parse(c.Listen)
}
log.Errorf("unsupported scheme %s", u.Scheme)
}
func (srv *tcpServer) run() {
l, err := net.Listen("tcp", srv.addr)
if err != nil { if err != nil {
log.Errorln(err) log.Fatalf("parse %s, error %s", c.Listen, err)
return
} }
defer l.Close()
for { switch u.Scheme {
conn, err := l.Accept() case "ws":
if err != nil { exists := false
log.Error(err) for i := 0; i < len(wsservers); i++ {
return if wsservers[i].addr == u.Host {
exists = true
wsservers[i].rule = append(wsservers[i].rule, forwardRule{u.Path, c.Remote})
break
} }
go srv.serve(conn)
} }
} if !exists {
wsservers = append(wsservers, wsServer{u.Host, []forwardRule{{u.Path, c.Remote}}})
func (srv *tcpServer) serve(c net.Conn) {
defer c.Close()
u, _ := url.Parse(srv.remote)
if u.Scheme == "ws" || u.Scheme == "wss" {
conn1, resp, err := dialer.Dial(srv.remote, nil)
if err != nil {
log.Errorln(err)
return
} }
resp.Body.Close() case "tcp":
if resp.StatusCode != http.StatusSwitchingProtocols { tcpservers = append(tcpservers, tcpServer{u.Host, c.Remote})
log.Errorf("dial remote ws %d", resp.StatusCode) default:
return log.Fatalf("unsupported scheme %s", u.Scheme)
} }
defer conn1.Close()
forwardWS2TCP(conn1, c)
return
} }
if u.Scheme == "tcp" { for _, srv := range wsservers {
conn1, err := net.Dial("tcp", u.Host) go srv.run()
if err != nil {
log.Errorln(err)
return
} }
defer conn1.Close()
forwardTCP2TCP(c, conn1) for _, srv := range tcpservers {
return go srv.run()
} }
log.Errorf("unsupported scheme %s", u.Scheme)
} }

@ -0,0 +1,69 @@
package main
import (
"net"
"net/http"
"net/url"
log "github.com/fangdingjun/go-log/v5"
)
type tcpServer struct {
addr string
remote string
}
func (srv *tcpServer) run() {
l, err := net.Listen("tcp", srv.addr)
if err != nil {
log.Errorln(err)
return
}
defer l.Close()
for {
conn, err := l.Accept()
if err != nil {
log.Error(err)
return
}
go srv.serve(conn)
}
}
func (srv *tcpServer) serve(c net.Conn) {
defer c.Close()
u, _ := url.Parse(srv.remote)
if u.Scheme == "ws" || u.Scheme == "wss" {
conn1, resp, err := dialer.Dial(srv.remote, nil)
if err != nil {
log.Errorln(err)
return
}
resp.Body.Close()
if resp.StatusCode != http.StatusSwitchingProtocols {
log.Errorf("dial remote ws %d", resp.StatusCode)
return
}
defer conn1.Close()
forwardWS2TCP(conn1, c)
return
}
if u.Scheme == "tcp" {
conn1, err := net.Dial("tcp", u.Host)
if err != nil {
log.Errorln(err)
return
}
defer conn1.Close()
forwardTCP2TCP(c, conn1)
return
}
log.Errorf("unsupported scheme %s", u.Scheme)
}

@ -0,0 +1,88 @@
package main
import (
"net"
"net/http"
"net/url"
log "github.com/fangdingjun/go-log/v5"
"github.com/gorilla/websocket"
)
var upgrader = websocket.Upgrader{
ReadBufferSize: 1024,
WriteBufferSize: 1024,
}
var dialer = &websocket.Dialer{}
type forwardRule struct {
local string
remote string
}
type wsServer struct {
addr string
rule []forwardRule
}
func (wss *wsServer) run() {
if err := http.ListenAndServe(wss.addr, wss); err != nil {
log.Errorln(err)
}
}
func (wss *wsServer) ServeHTTP(w http.ResponseWriter, r *http.Request) {
p := r.URL.Path
remote := ""
for _, ru := range wss.rule {
if ru.local == p {
remote = ru.remote
}
}
if remote == "" {
http.Error(w, "not found", http.StatusNotFound)
return
}
conn, err := upgrader.Upgrade(w, r, nil)
if err != nil {
log.Errorln(err)
http.Error(w, "bad request", http.StatusBadRequest)
return
}
defer conn.Close()
u, _ := url.Parse(remote)
if u.Scheme == "ws" || u.Scheme == "wss" {
conn1, resp, err := dialer.Dial(remote, nil)
if err != nil {
log.Errorln(err)
return
}
resp.Body.Close()
if resp.StatusCode != http.StatusSwitchingProtocols {
log.Errorf("dial remote ws %d", resp.StatusCode)
return
}
defer conn1.Close()
forwardWS2WS(conn, conn1)
return
}
if u.Scheme == "tcp" {
conn1, err := net.Dial("tcp", u.Host)
if err != nil {
log.Errorln(err)
return
}
defer conn1.Close()
forwardWS2TCP(conn, conn1)
return
}
log.Errorf("unsupported scheme %s", u.Scheme)
}
Loading…
Cancel
Save