You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
234 lines
3.9 KiB
Go
234 lines
3.9 KiB
Go
4 years ago
|
package main
|
||
|
|
||
|
import (
|
||
|
"io"
|
||
|
"net"
|
||
|
"net/http"
|
||
|
"net/url"
|
||
|
|
||
|
log "github.com/fangdingjun/go-log/v5"
|
||
|
"github.com/gorilla/websocket"
|
||
|
)
|
||
|
|
||
|
type forwardRule struct {
|
||
|
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)
|
||
|
go func() {
|
||
|
for {
|
||
|
t, data, err := conn.ReadMessage()
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
break
|
||
|
}
|
||
|
err = conn1.WriteMessage(t, data)
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
ch <- struct{}{}
|
||
|
}()
|
||
|
go func() {
|
||
|
for {
|
||
|
t, data, err := conn1.ReadMessage()
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
break
|
||
|
}
|
||
|
err = conn.WriteMessage(t, data)
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
ch <- struct{}{}
|
||
|
|
||
|
}()
|
||
|
<-ch
|
||
|
}
|
||
|
|
||
|
func forwardWS2TCP(conn1 *websocket.Conn, conn2 net.Conn) {
|
||
|
|
||
|
ch := make(chan struct{}, 2)
|
||
|
go func() {
|
||
|
for {
|
||
|
_, data, err := conn1.ReadMessage()
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
break
|
||
|
}
|
||
|
_, err = conn2.Write(data)
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
ch <- struct{}{}
|
||
|
}()
|
||
|
go func() {
|
||
|
buf := make([]byte, 1024)
|
||
|
for {
|
||
|
n, err := conn2.Read(buf)
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
break
|
||
|
}
|
||
|
err = conn1.WriteMessage(websocket.BinaryMessage, buf[:n])
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
ch <- struct{}{}
|
||
|
}()
|
||
|
<-ch
|
||
|
}
|
||
|
|
||
|
func forwardTCP2TCP(c1, c2 net.Conn) {
|
||
|
ch := make(chan struct{}, 2)
|
||
|
go func() {
|
||
|
_, err := io.Copy(c1, c2)
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
}
|
||
|
ch <- struct{}{}
|
||
|
}()
|
||
|
go func() {
|
||
|
_, err := io.Copy(c2, c1)
|
||
|
if err != nil {
|
||
|
log.Errorln(err)
|
||
|
}
|
||
|
ch <- struct{}{}
|
||
|
}()
|
||
|
<-ch
|
||
|
}
|
||
|
|
||
|
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()
|
||
|
defer conn1.Close()
|
||
|
if resp.StatusCode != http.StatusSwitchingProtocols {
|
||
|
log.Errorf("dial remote ws %d", resp.StatusCode)
|
||
|
return
|
||
|
}
|
||
|
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)
|
||
|
}
|
||
|
|
||
|
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()
|
||
|
defer conn1.Close()
|
||
|
if resp.StatusCode != http.StatusSwitchingProtocols {
|
||
|
log.Errorf("dial remote ws %d", resp.StatusCode)
|
||
|
return
|
||
|
}
|
||
|
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)
|
||
|
}
|