mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-19 21:11:32 +01:00
405 lines
11 KiB
Go
405 lines
11 KiB
Go
package remote
|
|
|
|
import (
|
|
"context"
|
|
"encoding/base64"
|
|
"errors"
|
|
"fmt"
|
|
"os/exec"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/scripthaus-dev/mshell/pkg/base"
|
|
"github.com/scripthaus-dev/mshell/pkg/packet"
|
|
"github.com/scripthaus-dev/mshell/pkg/shexec"
|
|
"github.com/scripthaus-dev/sh2-server/pkg/scpacket"
|
|
"github.com/scripthaus-dev/sh2-server/pkg/sstore"
|
|
)
|
|
|
|
const RemoteTypeMShell = "mshell"
|
|
const DefaultTermRows = 25
|
|
const DefaultTermCols = 80
|
|
const DefaultTerm = "xterm-256color"
|
|
|
|
const (
|
|
StatusInit = "init"
|
|
StatusConnected = "connected"
|
|
StatusDisconnected = "disconnected"
|
|
StatusError = "error"
|
|
)
|
|
|
|
var GlobalStore *Store
|
|
|
|
type Store struct {
|
|
Lock *sync.Mutex
|
|
Map map[string]*MShellProc // key=remoteid
|
|
CmdStatusCallback func(ck base.CommandKey, status string)
|
|
}
|
|
|
|
type RemoteState struct {
|
|
RemoteType string `json:"remotetype"`
|
|
RemoteId string `json:"remoteid"`
|
|
RemoteName string `json:"remotename"`
|
|
RemoteVars map[string]string `json:"remotevars"`
|
|
Status string `json:"status"`
|
|
DefaultState *sstore.RemoteState `json:"defaultstate"`
|
|
}
|
|
|
|
type MShellProc struct {
|
|
Lock *sync.Mutex
|
|
Remote *sstore.RemoteType
|
|
|
|
// runtime
|
|
Status string
|
|
ServerProc *shexec.ClientProc
|
|
Err error
|
|
|
|
RunningCmds []base.CommandKey
|
|
}
|
|
|
|
func LoadRemotes(ctx context.Context) error {
|
|
GlobalStore = &Store{
|
|
Lock: &sync.Mutex{},
|
|
Map: make(map[string]*MShellProc),
|
|
}
|
|
allRemotes, err := sstore.GetAllRemotes(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, remote := range allRemotes {
|
|
msh := MakeMShell(remote)
|
|
GlobalStore.Map[remote.RemoteId] = msh
|
|
if remote.AutoConnect {
|
|
go msh.Launch()
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func GetRemoteByName(name string) *MShellProc {
|
|
GlobalStore.Lock.Lock()
|
|
defer GlobalStore.Lock.Unlock()
|
|
for _, msh := range GlobalStore.Map {
|
|
if msh.Remote.RemoteName == name {
|
|
return msh
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func GetRemoteById(remoteId string) *MShellProc {
|
|
GlobalStore.Lock.Lock()
|
|
defer GlobalStore.Lock.Unlock()
|
|
return GlobalStore.Map[remoteId]
|
|
}
|
|
|
|
func GetAllRemoteState() []RemoteState {
|
|
GlobalStore.Lock.Lock()
|
|
defer GlobalStore.Lock.Unlock()
|
|
|
|
var rtn []RemoteState
|
|
for _, proc := range GlobalStore.Map {
|
|
state := RemoteState{
|
|
RemoteType: proc.Remote.RemoteType,
|
|
RemoteId: proc.Remote.RemoteId,
|
|
RemoteName: proc.Remote.RemoteName,
|
|
Status: proc.Status,
|
|
}
|
|
vars := make(map[string]string)
|
|
vars["user"], vars["host"] = proc.Remote.GetUserHost()
|
|
if proc.ServerProc != nil && proc.ServerProc.InitPk != nil {
|
|
state.DefaultState = &sstore.RemoteState{Cwd: proc.ServerProc.InitPk.HomeDir}
|
|
vars["home"] = proc.ServerProc.InitPk.HomeDir
|
|
vars["remoteuser"] = proc.ServerProc.InitPk.User
|
|
vars["remotehost"] = proc.ServerProc.InitPk.HostName
|
|
if proc.Remote.SSHOpts == nil || proc.Remote.SSHOpts.SSHHost == "" {
|
|
vars["local"] = "1"
|
|
}
|
|
}
|
|
state.RemoteVars = vars
|
|
rtn = append(rtn, state)
|
|
}
|
|
return rtn
|
|
}
|
|
|
|
func MakeMShell(r *sstore.RemoteType) *MShellProc {
|
|
rtn := &MShellProc{Lock: &sync.Mutex{}, Remote: r, Status: StatusInit}
|
|
return rtn
|
|
}
|
|
|
|
func (msh *MShellProc) Launch() {
|
|
msh.Lock.Lock()
|
|
defer msh.Lock.Unlock()
|
|
|
|
msPath, err := base.GetMShellPath()
|
|
if err != nil {
|
|
msh.Status = StatusError
|
|
msh.Err = err
|
|
return
|
|
}
|
|
ecmd := exec.Command(msPath, "--server")
|
|
cproc, err := shexec.MakeClientProc(ecmd)
|
|
if err != nil {
|
|
msh.Status = StatusError
|
|
msh.Err = err
|
|
return
|
|
}
|
|
msh.ServerProc = cproc
|
|
msh.Status = StatusConnected
|
|
go func() {
|
|
exitErr := cproc.Cmd.Wait()
|
|
exitCode := shexec.GetExitCode(exitErr)
|
|
msh.WithLock(func() {
|
|
if msh.Status == StatusConnected {
|
|
msh.Status = StatusDisconnected
|
|
}
|
|
})
|
|
|
|
fmt.Printf("[error] RUNNER PROC EXITED code[%d]\n", exitCode)
|
|
}()
|
|
go msh.ProcessPackets()
|
|
return
|
|
}
|
|
|
|
func (msh *MShellProc) IsConnected() bool {
|
|
msh.Lock.Lock()
|
|
defer msh.Lock.Unlock()
|
|
return msh.Status == StatusConnected
|
|
}
|
|
|
|
func (msh *MShellProc) IsCmdRunning(ck base.CommandKey) bool {
|
|
msh.Lock.Lock()
|
|
defer msh.Lock.Unlock()
|
|
for _, runningCk := range msh.RunningCmds {
|
|
if runningCk == ck {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (msh *MShellProc) SendInput(pk *packet.InputPacketType) error {
|
|
if !msh.IsConnected() {
|
|
return fmt.Errorf("remote is not connected, cannot send input")
|
|
}
|
|
if !msh.IsCmdRunning(pk.CK) {
|
|
return fmt.Errorf("cannot send input, cmd is not running")
|
|
}
|
|
dataPk := packet.MakeDataPacket()
|
|
dataPk.CK = pk.CK
|
|
dataPk.FdNum = 0 // stdin
|
|
dataPk.Data64 = pk.InputData64
|
|
return msh.ServerProc.Input.SendPacket(dataPk)
|
|
}
|
|
|
|
func convertRemoteState(rs scpacket.RemoteState) sstore.RemoteState {
|
|
return sstore.RemoteState{Cwd: rs.Cwd}
|
|
}
|
|
|
|
func makeTermOpts() sstore.TermOpts {
|
|
return sstore.TermOpts{Rows: DefaultTermRows, Cols: DefaultTermCols, FlexRows: true}
|
|
}
|
|
|
|
func RunCommand(ctx context.Context, pk *scpacket.FeCommandPacketType, cmdId string) (*sstore.CmdType, error) {
|
|
msh := GetRemoteById(pk.RemoteState.RemoteId)
|
|
if msh == nil {
|
|
return nil, fmt.Errorf("no remote id=%s found", pk.RemoteState.RemoteId)
|
|
}
|
|
if !msh.IsConnected() {
|
|
return nil, fmt.Errorf("remote '%s' is not connected", msh.Remote.RemoteName)
|
|
}
|
|
runPacket := packet.MakeRunPacket()
|
|
runPacket.ReqId = uuid.New().String()
|
|
runPacket.CK = base.MakeCommandKey(pk.SessionId, cmdId)
|
|
runPacket.Cwd = pk.RemoteState.Cwd
|
|
runPacket.Env = nil
|
|
runPacket.UsePty = true
|
|
runPacket.TermOpts = &packet.TermOpts{Rows: DefaultTermRows, Cols: DefaultTermCols, Term: DefaultTerm}
|
|
runPacket.Command = strings.TrimSpace(pk.CmdStr)
|
|
fmt.Printf("RUN-CMD> %s reqid=%s (msh=%v)\n", runPacket.CK, runPacket.ReqId, msh.Remote)
|
|
msh.ServerProc.Output.RegisterRpc(runPacket.ReqId)
|
|
err := shexec.SendRunPacketAndRunData(ctx, msh.ServerProc.Input, runPacket)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("sending run packet to remote: %w", err)
|
|
}
|
|
rtnPk := msh.ServerProc.Output.WaitForResponse(ctx, runPacket.ReqId)
|
|
startPk, ok := rtnPk.(*packet.CmdStartPacketType)
|
|
if !ok {
|
|
respPk, ok := rtnPk.(*packet.ResponsePacketType)
|
|
if !ok {
|
|
return nil, fmt.Errorf("invalid response received from server for run packet: %s", packet.AsString(rtnPk))
|
|
}
|
|
if respPk.Error != "" {
|
|
return nil, errors.New(respPk.Error)
|
|
}
|
|
return nil, fmt.Errorf("invalid response received from server for run packet: %s", packet.AsString(rtnPk))
|
|
}
|
|
status := sstore.CmdStatusRunning
|
|
if runPacket.Detached {
|
|
status = sstore.CmdStatusDetached
|
|
}
|
|
cmd := &sstore.CmdType{
|
|
SessionId: pk.SessionId,
|
|
CmdId: startPk.CK.GetCmdId(),
|
|
CmdStr: runPacket.Command,
|
|
RemoteId: msh.Remote.RemoteId,
|
|
RemoteState: convertRemoteState(pk.RemoteState),
|
|
TermOpts: makeTermOpts(),
|
|
Status: status,
|
|
StartPk: startPk,
|
|
DonePk: nil,
|
|
RunOut: nil,
|
|
}
|
|
err = sstore.AppendToCmdPtyBlob(ctx, cmd.SessionId, cmd.CmdId, nil, sstore.PosAppend)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
msh.AddRunningCmd(startPk.CK)
|
|
return cmd, nil
|
|
}
|
|
|
|
func (msh *MShellProc) AddRunningCmd(ck base.CommandKey) {
|
|
msh.Lock.Lock()
|
|
defer msh.Lock.Unlock()
|
|
msh.RunningCmds = append(msh.RunningCmds, ck)
|
|
}
|
|
|
|
func (msh *MShellProc) PacketRpc(ctx context.Context, pk packet.RpcPacketType) (*packet.ResponsePacketType, error) {
|
|
if !msh.IsConnected() {
|
|
return nil, fmt.Errorf("runner is not connected")
|
|
}
|
|
if pk == nil {
|
|
return nil, fmt.Errorf("PacketRpc passed nil packet")
|
|
}
|
|
reqId := pk.GetReqId()
|
|
msh.ServerProc.Output.RegisterRpc(reqId)
|
|
defer msh.ServerProc.Output.UnRegisterRpc(reqId)
|
|
err := msh.ServerProc.Input.SendPacketCtx(ctx, pk)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
rtnPk := msh.ServerProc.Output.WaitForResponse(ctx, reqId)
|
|
if rtnPk == nil {
|
|
return nil, ctx.Err()
|
|
}
|
|
if respPk, ok := rtnPk.(*packet.ResponsePacketType); ok {
|
|
return respPk, nil
|
|
}
|
|
return nil, fmt.Errorf("invalid response packet received: %s", packet.AsString(rtnPk))
|
|
}
|
|
|
|
func (runner *MShellProc) WithLock(fn func()) {
|
|
runner.Lock.Lock()
|
|
defer runner.Lock.Unlock()
|
|
fn()
|
|
}
|
|
|
|
func makeDataAckPacket(ck base.CommandKey, fdNum int, ackLen int, err error) *packet.DataAckPacketType {
|
|
ack := packet.MakeDataAckPacket()
|
|
ack.CK = ck
|
|
ack.FdNum = fdNum
|
|
ack.AckLen = ackLen
|
|
if err != nil {
|
|
ack.Error = err.Error()
|
|
}
|
|
return ack
|
|
}
|
|
|
|
func (msh *MShellProc) handleCmdDonePacket(donePk *packet.CmdDonePacketType) {
|
|
err := sstore.UpdateCmdDonePk(context.Background(), donePk)
|
|
if err != nil {
|
|
fmt.Printf("[error] updating cmddone: %v\n", err)
|
|
return
|
|
}
|
|
if GlobalStore.CmdStatusCallback != nil {
|
|
GlobalStore.CmdStatusCallback(donePk.CK, sstore.CmdStatusDone)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (msh *MShellProc) handleCmdErrorPacket(errPk *packet.CmdErrorPacketType) {
|
|
err := sstore.AppendCmdErrorPk(context.Background(), errPk)
|
|
if err != nil {
|
|
fmt.Printf("[error] adding cmderr: %v\n", err)
|
|
return
|
|
}
|
|
return
|
|
}
|
|
|
|
func (msh *MShellProc) notifyHangups_nolock() {
|
|
if GlobalStore.CmdStatusCallback != nil {
|
|
for _, ck := range msh.RunningCmds {
|
|
GlobalStore.CmdStatusCallback(ck, sstore.CmdStatusHangup)
|
|
}
|
|
}
|
|
msh.RunningCmds = nil
|
|
}
|
|
|
|
func (runner *MShellProc) ProcessPackets() {
|
|
defer runner.WithLock(func() {
|
|
if runner.Status == StatusConnected {
|
|
runner.Status = StatusDisconnected
|
|
}
|
|
err := sstore.HangupRunningCmdsByRemoteId(context.Background(), runner.Remote.RemoteId)
|
|
if err != nil {
|
|
fmt.Printf("[error] calling HUP on remoteid=%d cmds\n", runner.Remote.RemoteId)
|
|
}
|
|
runner.notifyHangups_nolock()
|
|
})
|
|
for pk := range runner.ServerProc.Output.MainCh {
|
|
if pk.GetType() == packet.DataPacketStr {
|
|
dataPk := pk.(*packet.DataPacketType)
|
|
realData, err := base64.StdEncoding.DecodeString(dataPk.Data64)
|
|
if err != nil {
|
|
ack := makeDataAckPacket(dataPk.CK, dataPk.FdNum, 0, err)
|
|
runner.ServerProc.Input.SendPacket(ack)
|
|
continue
|
|
}
|
|
var ack *packet.DataAckPacketType
|
|
if len(realData) > 0 {
|
|
err = sstore.AppendToCmdPtyBlob(context.Background(), dataPk.CK.GetSessionId(), dataPk.CK.GetCmdId(), realData, sstore.PosAppend)
|
|
if err != nil {
|
|
ack = makeDataAckPacket(dataPk.CK, dataPk.FdNum, 0, err)
|
|
} else {
|
|
ack = makeDataAckPacket(dataPk.CK, dataPk.FdNum, len(realData), nil)
|
|
}
|
|
}
|
|
if ack != nil {
|
|
runner.ServerProc.Input.SendPacket(ack)
|
|
}
|
|
// fmt.Printf("data %s fd=%d len=%d eof=%v err=%v\n", dataPk.CK, dataPk.FdNum, len(realData), dataPk.Eof, dataPk.Error)
|
|
continue
|
|
}
|
|
if pk.GetType() == packet.CmdDataPacketStr {
|
|
dataPacket := pk.(*packet.CmdDataPacketType)
|
|
fmt.Printf("cmd-data %s pty=%d run=%d\n", dataPacket.CK, dataPacket.PtyDataLen, dataPacket.RunDataLen)
|
|
continue
|
|
}
|
|
if pk.GetType() == packet.CmdDonePacketStr {
|
|
runner.handleCmdDonePacket(pk.(*packet.CmdDonePacketType))
|
|
continue
|
|
}
|
|
if pk.GetType() == packet.CmdErrorPacketStr {
|
|
runner.handleCmdErrorPacket(pk.(*packet.CmdErrorPacketType))
|
|
continue
|
|
}
|
|
if pk.GetType() == packet.MessagePacketStr {
|
|
msgPacket := pk.(*packet.MessagePacketType)
|
|
fmt.Printf("# %s\n", msgPacket.Message)
|
|
continue
|
|
}
|
|
if pk.GetType() == packet.RawPacketStr {
|
|
rawPacket := pk.(*packet.RawPacketType)
|
|
fmt.Printf("stderr> %s\n", rawPacket.Data)
|
|
continue
|
|
}
|
|
if pk.GetType() == packet.CmdStartPacketStr {
|
|
startPk := pk.(*packet.CmdStartPacketType)
|
|
fmt.Printf("start> reqid=%s (%p)\n", startPk.RespId, runner.ServerProc.Output)
|
|
continue
|
|
}
|
|
fmt.Printf("MSH> %s\n", packet.AsString(pk))
|
|
}
|
|
}
|