Files
natpass/code/client/shell/http.go
2021-09-27 18:28:26 +08:00

83 lines
1.9 KiB
Go

package shell
import (
"natpass/code/client/pool"
"natpass/code/network"
"natpass/code/utils"
"net/http"
"sync"
"github.com/gorilla/websocket"
"github.com/lwch/logging"
"github.com/lwch/runtime"
)
var upgrader = websocket.Upgrader{}
// WS websocket for shell
func (shell *Shell) WS(pool *pool.Pool, w http.ResponseWriter, r *http.Request) {
id, err := runtime.UUID(16, "0123456789abcdef")
if err != nil {
logging.Error("failed to generate link_id for shell: %s, err=%v",
shell.Name, err)
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
conn := pool.Get(id)
conn.SendShellCreate(id, shell.cfg)
local, err := upgrader.Upgrade(w, r, nil)
if err != nil {
logging.Error("upgrade websocket failed: %s, err=%v", shell.Name, err)
http.Error(w, err.Error(), http.StatusServiceUnavailable)
return
}
defer local.Close()
var wg sync.WaitGroup
wg.Add(2)
go func() {
defer wg.Done()
shell.localForward(id, local, conn)
}()
go func() {
defer wg.Done()
shell.remoteForward(id, conn.ChanRead(id), local)
}()
wg.Wait()
}
func (shell *Shell) localForward(id string, local *websocket.Conn, remote *pool.Conn) {
defer utils.Recover("localForward")
defer local.Close()
for {
_, data, err := local.ReadMessage()
if err != nil {
// TODO: close
logging.Error("read local data for %s failed: %v", shell.Name, err)
return
}
remote.SendShellData(shell.cfg.Target, remote.Idx, id, data)
}
}
func (shell *Shell) remoteForward(id string, ch <-chan *network.Msg, local *websocket.Conn) {
defer utils.Recover("remoteForward")
defer local.Close()
for {
msg := <-ch
if msg == nil {
return
}
switch msg.GetXType() {
case network.Msg_shell_data:
err := local.WriteMessage(websocket.TextMessage, msg.GetSdata().GetData())
if err != nil {
logging.Error("write data for %s failed: %v", shell.Name, err)
return
}
// TODO: other
}
}
}