mirror of
https://github.com/lwch/natpass
synced 2025-10-13 00:33:41 +08:00
83 lines
1.9 KiB
Go
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
|
|
}
|
|
}
|
|
}
|