2022-06-10 09:35:24 +02:00
|
|
|
// Copyright 2022 Dashborg Inc
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
// file, You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
2022-06-18 03:11:49 +02:00
|
|
|
"os/user"
|
2022-06-23 21:48:45 +02:00
|
|
|
"strings"
|
2022-06-10 09:35:24 +02:00
|
|
|
"syscall"
|
2022-06-11 06:37:21 +02:00
|
|
|
"time"
|
2022-06-10 09:35:24 +02:00
|
|
|
|
2022-06-23 19:16:54 +02:00
|
|
|
"github.com/scripthaus-dev/mshell/pkg/base"
|
|
|
|
"github.com/scripthaus-dev/mshell/pkg/cmdtail"
|
|
|
|
"github.com/scripthaus-dev/mshell/pkg/packet"
|
|
|
|
"github.com/scripthaus-dev/mshell/pkg/shexec"
|
2022-06-25 08:42:00 +02:00
|
|
|
"golang.org/x/sys/unix"
|
2022-06-10 09:35:24 +02:00
|
|
|
)
|
|
|
|
|
2022-06-23 21:48:45 +02:00
|
|
|
const MShellVersion = "0.1.0"
|
|
|
|
|
|
|
|
// in single run mode, we don't want mshell to die from signals
|
|
|
|
// since we want the single mshell to persist even if session / main mshell
|
2022-06-11 06:37:21 +02:00
|
|
|
// is terminated.
|
|
|
|
func setupSingleSignals(cmd *shexec.ShExecType) {
|
2022-06-10 09:35:24 +02:00
|
|
|
sigCh := make(chan os.Signal, 1)
|
2022-06-11 06:37:21 +02:00
|
|
|
signal.Notify(sigCh, syscall.SIGINT, syscall.SIGTERM, syscall.SIGHUP)
|
2022-06-10 09:35:24 +02:00
|
|
|
go func() {
|
2022-06-11 06:37:21 +02:00
|
|
|
for range sigCh {
|
|
|
|
// do nothing
|
2022-06-10 09:35:24 +02:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2022-06-27 21:03:47 +02:00
|
|
|
func doSingle(ck base.CommandKey) {
|
2022-06-27 21:14:07 +02:00
|
|
|
packetParser := packet.MakePacketParser(os.Stdin)
|
2022-06-11 06:37:21 +02:00
|
|
|
sender := packet.MakePacketSender(os.Stdout)
|
2022-06-10 09:35:24 +02:00
|
|
|
var runPacket *packet.RunPacketType
|
2022-06-27 21:14:07 +02:00
|
|
|
for pk := range packetParser.MainCh {
|
2022-06-10 09:35:24 +02:00
|
|
|
if pk.GetType() == packet.PingPacketStr {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if pk.GetType() == packet.RunPacketStr {
|
|
|
|
runPacket, _ = pk.(*packet.RunPacketType)
|
|
|
|
break
|
|
|
|
}
|
2022-06-23 21:48:45 +02:00
|
|
|
sender.SendErrorPacket(fmt.Sprintf("invalid packet '%s' sent to mshell", pk.GetType()))
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
if runPacket == nil {
|
|
|
|
sender.SendErrorPacket("did not receive a 'run' packet")
|
|
|
|
return
|
|
|
|
}
|
2022-06-27 21:03:47 +02:00
|
|
|
if runPacket.CK.IsEmpty() {
|
|
|
|
runPacket.CK = ck
|
2022-06-11 06:37:21 +02:00
|
|
|
}
|
2022-06-27 21:03:47 +02:00
|
|
|
if runPacket.CK != ck {
|
|
|
|
sender.SendErrorPacket(fmt.Sprintf("run packet cmdid[%s] did not match arg[%s]", runPacket.CK, ck))
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
cmd, err := shexec.RunCommand(runPacket, sender)
|
|
|
|
if err != nil {
|
|
|
|
sender.SendErrorPacket(fmt.Sprintf("error running command: %v", err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
setupSingleSignals(cmd)
|
2022-06-23 21:48:45 +02:00
|
|
|
startPacket := cmd.MakeCmdStartPacket()
|
2022-06-11 06:37:21 +02:00
|
|
|
sender.SendPacket(startPacket)
|
2022-06-23 21:48:45 +02:00
|
|
|
donePacket := cmd.WaitForCommand()
|
2022-06-11 06:37:21 +02:00
|
|
|
sender.SendPacket(donePacket)
|
|
|
|
sender.CloseSendCh()
|
|
|
|
sender.WaitForDone()
|
|
|
|
}
|
|
|
|
|
|
|
|
func doMainRun(pk *packet.RunPacketType, sender *packet.PacketSender) {
|
|
|
|
err := shexec.ValidateRunPacket(pk)
|
|
|
|
if err != nil {
|
2022-06-27 21:03:47 +02:00
|
|
|
sender.SendPacket(packet.MakeCKErrorPacket(pk.CK, fmt.Sprintf("invalid run packet: %v", err)))
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
2022-06-27 21:03:47 +02:00
|
|
|
fileNames, err := base.GetCommandFileNames(pk.CK)
|
2022-06-11 06:37:21 +02:00
|
|
|
if err != nil {
|
2022-06-27 21:03:47 +02:00
|
|
|
sender.SendPacket(packet.MakeCKErrorPacket(pk.CK, fmt.Sprintf("cannot get command file names: %v", err)))
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
2022-06-27 21:03:47 +02:00
|
|
|
cmd, err := shexec.MakeRunnerExec(pk.CK)
|
2022-06-11 06:37:21 +02:00
|
|
|
if err != nil {
|
2022-06-27 21:03:47 +02:00
|
|
|
sender.SendPacket(packet.MakeCKErrorPacket(pk.CK, fmt.Sprintf("cannot make mshell command: %v", err)))
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
cmdStdin, err := cmd.StdinPipe()
|
|
|
|
if err != nil {
|
2022-06-27 21:03:47 +02:00
|
|
|
sender.SendPacket(packet.MakeCKErrorPacket(pk.CK, fmt.Sprintf("cannot pipe stdin to command: %v", err)))
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
2022-06-17 21:27:29 +02:00
|
|
|
// touch ptyout file (should exist for tailer to work correctly)
|
|
|
|
ptyOutFd, err := os.OpenFile(fileNames.PtyOutFile, os.O_CREATE|os.O_TRUNC|os.O_APPEND|os.O_WRONLY, 0600)
|
|
|
|
if err != nil {
|
2022-06-27 21:03:47 +02:00
|
|
|
sender.SendPacket(packet.MakeCKErrorPacket(pk.CK, fmt.Sprintf("cannot open pty out file '%s': %v", fileNames.PtyOutFile, err)))
|
2022-06-17 21:27:29 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
ptyOutFd.Close() // just opened to create the file, can close right after
|
2022-06-11 06:37:21 +02:00
|
|
|
runnerOutFd, err := os.OpenFile(fileNames.RunnerOutFile, os.O_CREATE|os.O_TRUNC|os.O_APPEND|os.O_WRONLY, 0600)
|
|
|
|
if err != nil {
|
2022-06-27 21:03:47 +02:00
|
|
|
sender.SendPacket(packet.MakeCKErrorPacket(pk.CK, fmt.Sprintf("cannot open runner out file '%s': %v", fileNames.RunnerOutFile, err)))
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
2022-06-17 20:29:33 +02:00
|
|
|
defer runnerOutFd.Close()
|
2022-06-11 06:37:21 +02:00
|
|
|
cmd.Stdout = runnerOutFd
|
|
|
|
cmd.Stderr = runnerOutFd
|
|
|
|
err = cmd.Start()
|
|
|
|
if err != nil {
|
2022-06-27 21:03:47 +02:00
|
|
|
sender.SendPacket(packet.MakeCKErrorPacket(pk.CK, fmt.Sprintf("error starting command: %v", err)))
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
err = packet.SendPacket(cmdStdin, pk)
|
|
|
|
if err != nil {
|
2022-06-27 21:03:47 +02:00
|
|
|
sender.SendPacket(packet.MakeCKErrorPacket(pk.CK, fmt.Sprintf("error sending forked runner command: %v", err)))
|
2022-06-10 09:35:24 +02:00
|
|
|
return
|
|
|
|
}
|
2022-06-11 06:37:21 +02:00
|
|
|
cmdStdin.Close()
|
|
|
|
|
|
|
|
// clean up zombies
|
|
|
|
cmd.Wait()
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2022-06-15 07:16:58 +02:00
|
|
|
func doGetCmd(tailer *cmdtail.Tailer, pk *packet.GetCmdPacketType, sender *packet.PacketSender) error {
|
|
|
|
err := tailer.AddWatch(pk)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-06-11 06:37:21 +02:00
|
|
|
func doMain() {
|
2022-06-15 07:16:58 +02:00
|
|
|
scHomeDir, err := base.GetScHomeDir()
|
2022-06-11 06:37:21 +02:00
|
|
|
if err != nil {
|
|
|
|
packet.SendErrorPacket(os.Stdout, err.Error())
|
2022-06-10 09:35:24 +02:00
|
|
|
return
|
|
|
|
}
|
2022-06-15 07:16:58 +02:00
|
|
|
homeDir := base.GetHomeDir()
|
2022-06-11 06:37:21 +02:00
|
|
|
err = os.Chdir(homeDir)
|
|
|
|
if err != nil {
|
2022-06-15 07:16:58 +02:00
|
|
|
packet.SendErrorPacket(os.Stdout, fmt.Sprintf("cannot change directory to $HOME '%s': %v", homeDir, err))
|
2022-06-10 09:35:24 +02:00
|
|
|
return
|
|
|
|
}
|
2022-06-23 21:48:45 +02:00
|
|
|
_, err = base.GetMShellPath()
|
2022-06-10 09:35:24 +02:00
|
|
|
if err != nil {
|
2022-06-11 06:37:21 +02:00
|
|
|
packet.SendErrorPacket(os.Stdout, err.Error())
|
|
|
|
return
|
|
|
|
}
|
2022-06-27 21:14:07 +02:00
|
|
|
packetParser := packet.MakePacketParser(os.Stdin)
|
2022-06-11 06:37:21 +02:00
|
|
|
sender := packet.MakePacketSender(os.Stdout)
|
2022-06-16 10:10:56 +02:00
|
|
|
tailer, err := cmdtail.MakeTailer(sender.SendCh)
|
2022-06-15 07:16:58 +02:00
|
|
|
if err != nil {
|
|
|
|
packet.SendErrorPacket(os.Stdout, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
go tailer.Run()
|
2022-06-23 21:48:45 +02:00
|
|
|
initPacket := packet.MakeInitPacket()
|
2022-06-15 07:16:58 +02:00
|
|
|
initPacket.Env = os.Environ()
|
|
|
|
initPacket.HomeDir = homeDir
|
|
|
|
initPacket.ScHomeDir = scHomeDir
|
2022-06-18 03:11:49 +02:00
|
|
|
if user, _ := user.Current(); user != nil {
|
|
|
|
initPacket.User = user.Username
|
|
|
|
}
|
2022-06-15 07:16:58 +02:00
|
|
|
sender.SendPacket(initPacket)
|
2022-06-27 21:14:07 +02:00
|
|
|
for pk := range packetParser.MainCh {
|
2022-06-11 06:37:21 +02:00
|
|
|
if pk.GetType() == packet.PingPacketStr {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if pk.GetType() == packet.RunPacketStr {
|
|
|
|
doMainRun(pk.(*packet.RunPacketType), sender)
|
|
|
|
continue
|
|
|
|
}
|
2022-06-15 07:16:58 +02:00
|
|
|
if pk.GetType() == packet.GetCmdPacketStr {
|
|
|
|
err = doGetCmd(tailer, pk.(*packet.GetCmdPacketType), sender)
|
|
|
|
if err != nil {
|
|
|
|
errPk := packet.MakeErrorPacket(err.Error())
|
|
|
|
sender.SendPacket(errPk)
|
2022-06-23 18:56:54 +02:00
|
|
|
continue
|
2022-06-15 07:16:58 +02:00
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
2022-06-23 18:56:54 +02:00
|
|
|
if pk.GetType() == packet.CdPacketStr {
|
|
|
|
cdPacket := pk.(*packet.CdPacketType)
|
|
|
|
err := os.Chdir(cdPacket.Dir)
|
|
|
|
resp := packet.MakeResponsePacket(cdPacket.PacketId)
|
|
|
|
if err != nil {
|
|
|
|
resp.Error = err.Error()
|
|
|
|
} else {
|
|
|
|
resp.Success = true
|
|
|
|
}
|
|
|
|
sender.SendPacket(resp)
|
|
|
|
continue
|
|
|
|
}
|
2022-06-11 06:37:21 +02:00
|
|
|
if pk.GetType() == packet.ErrorPacketStr {
|
|
|
|
errPk := pk.(*packet.ErrorPacketType)
|
2022-06-23 21:48:45 +02:00
|
|
|
errPk.Error = "invalid packet sent to mshell: " + errPk.Error
|
2022-06-11 06:37:21 +02:00
|
|
|
sender.SendPacket(errPk)
|
|
|
|
continue
|
|
|
|
}
|
2022-06-23 21:48:45 +02:00
|
|
|
sender.SendErrorPacket(fmt.Sprintf("invalid packet '%s' sent to mshell", pk.GetType()))
|
2022-06-11 06:37:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-23 21:48:45 +02:00
|
|
|
func handleRemote() {
|
2022-06-27 21:14:07 +02:00
|
|
|
packetParser := packet.MakePacketParser(os.Stdin)
|
2022-06-23 21:48:45 +02:00
|
|
|
sender := packet.MakePacketSender(os.Stdout)
|
|
|
|
defer func() {
|
|
|
|
// wait for sender to complete
|
|
|
|
close(sender.SendCh)
|
|
|
|
<-sender.DoneCh
|
|
|
|
}()
|
|
|
|
initPacket := packet.MakeInitPacket()
|
|
|
|
initPacket.Version = MShellVersion
|
|
|
|
sender.SendPacket(initPacket)
|
|
|
|
var runPacket *packet.RunPacketType
|
2022-06-27 21:14:07 +02:00
|
|
|
for pk := range packetParser.MainCh {
|
2022-06-23 21:48:45 +02:00
|
|
|
if pk.GetType() == packet.PingPacketStr {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if pk.GetType() == packet.RunPacketStr {
|
|
|
|
runPacket, _ = pk.(*packet.RunPacketType)
|
|
|
|
break
|
|
|
|
}
|
2022-06-24 09:02:18 +02:00
|
|
|
if pk.GetType() == packet.RawPacketStr {
|
|
|
|
rawPk := pk.(*packet.RawPacketType)
|
|
|
|
sender.SendMessage("got raw packet '%s'", rawPk.Data)
|
|
|
|
continue
|
|
|
|
}
|
2022-06-23 21:48:45 +02:00
|
|
|
sender.SendErrorPacket(fmt.Sprintf("invalid packet '%s' sent to mshell", pk.GetType()))
|
|
|
|
return
|
|
|
|
}
|
2022-06-25 09:30:41 +02:00
|
|
|
if runPacket == nil {
|
|
|
|
sender.SendErrorPacket(fmt.Sprintf("no run packet received"))
|
|
|
|
return
|
|
|
|
}
|
2022-06-23 21:48:45 +02:00
|
|
|
cmd, err := shexec.RunCommand(runPacket, sender)
|
|
|
|
if err != nil {
|
|
|
|
sender.SendErrorPacket(fmt.Sprintf("error running command: %v", err))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer cmd.Close()
|
|
|
|
startPacket := cmd.MakeCmdStartPacket()
|
|
|
|
sender.SendPacket(startPacket)
|
2022-06-27 21:14:07 +02:00
|
|
|
cmd.RunRemoteIOAndWait(packetParser, sender)
|
2022-06-23 21:48:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func handleServer() {
|
|
|
|
}
|
|
|
|
|
2022-06-25 08:42:00 +02:00
|
|
|
func detectOpenFds() ([]packet.RemoteFd, error) {
|
|
|
|
var fds []packet.RemoteFd
|
|
|
|
for fdNum := 3; fdNum <= 64; fdNum++ {
|
|
|
|
flags, err := unix.FcntlInt(uintptr(fdNum), unix.F_GETFL, 0)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
flags = flags & 3
|
|
|
|
rfd := packet.RemoteFd{FdNum: fdNum}
|
|
|
|
if flags&2 == 2 {
|
|
|
|
return nil, fmt.Errorf("invalid fd=%d, mshell does not support fds open for reading and writing", fdNum)
|
|
|
|
}
|
|
|
|
if flags&1 == 1 {
|
|
|
|
rfd.Write = true
|
|
|
|
} else {
|
|
|
|
rfd.Read = true
|
|
|
|
}
|
|
|
|
fds = append(fds, rfd)
|
|
|
|
}
|
|
|
|
return fds, nil
|
2022-06-24 09:02:18 +02:00
|
|
|
}
|
|
|
|
|
2022-06-24 22:25:09 +02:00
|
|
|
func parseClientOpts() (*shexec.ClientOpts, error) {
|
|
|
|
opts := &shexec.ClientOpts{}
|
2022-06-24 09:02:18 +02:00
|
|
|
iter := base.MakeOptsIter(os.Args[1:])
|
|
|
|
for iter.HasNext() {
|
|
|
|
argStr := iter.Next()
|
|
|
|
if argStr == "--ssh" {
|
|
|
|
if opts.IsSSH {
|
|
|
|
return nil, fmt.Errorf("duplicate '--ssh' option")
|
|
|
|
}
|
|
|
|
opts.IsSSH = true
|
|
|
|
break
|
|
|
|
}
|
2022-06-25 08:42:00 +02:00
|
|
|
if argStr == "--cwd" {
|
|
|
|
if !iter.HasNext() {
|
|
|
|
return nil, fmt.Errorf("'--cwd [dir]' missing directory")
|
|
|
|
}
|
|
|
|
opts.Cwd = iter.Next()
|
|
|
|
continue
|
|
|
|
}
|
2022-06-25 09:22:03 +02:00
|
|
|
if argStr == "--debug" {
|
|
|
|
opts.Debug = true
|
|
|
|
continue
|
|
|
|
}
|
2022-06-26 10:41:58 +02:00
|
|
|
if argStr == "--sudo" {
|
|
|
|
opts.Sudo = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if argStr == "--sudo-with-password" {
|
|
|
|
if !iter.HasNext() {
|
|
|
|
return nil, fmt.Errorf("'--sudo-with-password [pw]', missing password")
|
|
|
|
}
|
|
|
|
opts.Sudo = true
|
|
|
|
opts.SudoWithPass = true
|
|
|
|
opts.SudoPw = iter.Next()
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if argStr == "--sudo-with-passfile" {
|
|
|
|
if !iter.HasNext() {
|
|
|
|
return nil, fmt.Errorf("'--sudo-with-passfile [file]', missing file")
|
|
|
|
}
|
|
|
|
opts.Sudo = true
|
|
|
|
opts.SudoWithPass = true
|
|
|
|
fileName := iter.Next()
|
|
|
|
contents, err := os.ReadFile(fileName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("cannot read --sudo-with-passfile file '%s': %w", fileName, err)
|
|
|
|
}
|
|
|
|
opts.SudoPw = string(contents)
|
|
|
|
continue
|
|
|
|
}
|
2022-06-24 09:02:18 +02:00
|
|
|
}
|
|
|
|
if opts.IsSSH {
|
|
|
|
// parse SSH opts
|
|
|
|
for iter.HasNext() {
|
|
|
|
argStr := iter.Next()
|
|
|
|
if argStr == "--" {
|
|
|
|
opts.SSHOptsTerm = true
|
|
|
|
break
|
|
|
|
}
|
2022-06-25 09:30:41 +02:00
|
|
|
if argStr == "-t" || argStr == "-tt" {
|
|
|
|
return nil, fmt.Errorf("mshell cannot run over ssh -t")
|
|
|
|
}
|
2022-06-24 09:02:18 +02:00
|
|
|
opts.SSHOpts = append(opts.SSHOpts, argStr)
|
|
|
|
}
|
|
|
|
if !opts.SSHOptsTerm {
|
|
|
|
return nil, fmt.Errorf("ssh options must be terminated with '--' followed by [command]")
|
|
|
|
}
|
|
|
|
if !iter.HasNext() {
|
|
|
|
return nil, fmt.Errorf("no command specified")
|
|
|
|
}
|
|
|
|
opts.Command = strings.Join(iter.Rest(), " ")
|
|
|
|
if strings.TrimSpace(opts.Command) == "" {
|
|
|
|
return nil, fmt.Errorf("no command or empty command specified")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return opts, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func handleClient() (int, error) {
|
|
|
|
opts, err := parseClientOpts()
|
|
|
|
if err != nil {
|
|
|
|
return 1, fmt.Errorf("parsing opts: %w", err)
|
|
|
|
}
|
2022-06-26 10:41:58 +02:00
|
|
|
if opts.Debug {
|
|
|
|
packet.GlobalDebug = true
|
|
|
|
}
|
2022-06-24 09:02:18 +02:00
|
|
|
if !opts.IsSSH {
|
|
|
|
return 1, fmt.Errorf("when running in client mode '--ssh' option must be present")
|
|
|
|
}
|
2022-06-25 08:42:00 +02:00
|
|
|
fds, err := detectOpenFds()
|
|
|
|
if err != nil {
|
|
|
|
return 1, err
|
|
|
|
}
|
|
|
|
opts.Fds = fds
|
2022-06-24 22:25:09 +02:00
|
|
|
donePacket, err := shexec.RunClientSSHCommandAndWait(opts)
|
2022-06-24 09:02:18 +02:00
|
|
|
if err != nil {
|
2022-06-24 22:25:09 +02:00
|
|
|
return 1, err
|
2022-06-24 09:02:18 +02:00
|
|
|
}
|
2022-06-24 22:25:09 +02:00
|
|
|
return donePacket.ExitCode, nil
|
2022-06-23 21:48:45 +02:00
|
|
|
}
|
|
|
|
|
2022-06-24 09:02:18 +02:00
|
|
|
func handleUsage() {
|
2022-06-23 21:48:45 +02:00
|
|
|
usage := `
|
2022-06-24 09:02:18 +02:00
|
|
|
Client Usage: mshell [mshell-opts] --ssh [ssh-opts] user@host -- [command]
|
2022-06-23 21:48:45 +02:00
|
|
|
|
|
|
|
mshell multiplexes input and output streams to a remote command over ssh.
|
|
|
|
|
|
|
|
Options:
|
|
|
|
--cwd [dir] - execute remote command in [dir]
|
2022-06-24 09:02:18 +02:00
|
|
|
[command] - a single argument (should be quoted)
|
|
|
|
|
2022-06-26 10:53:07 +02:00
|
|
|
Sudo Options:
|
|
|
|
--sudo
|
|
|
|
--sudo-with-password [pw] (not recommended, use --sudo-with-passfile if possible)
|
|
|
|
--sudo-with-passfile [file]
|
|
|
|
|
|
|
|
Sudo options allow you to run the given command using "sudo". The first
|
|
|
|
option only works when you can sudo without a password. Your password will be passed
|
|
|
|
securely through a high numbered fd to "sudo -S". See full documentation for more details.
|
|
|
|
|
2022-06-24 09:02:18 +02:00
|
|
|
Examples:
|
|
|
|
# execute a python script remotely, with stdin still hooked up correctly
|
2022-06-26 10:53:07 +02:00
|
|
|
mshell --cwd "~/work" --ssh -i key.pem ubuntu@somehost -- "python3 /dev/fd/4" 4< myscript.py
|
2022-06-24 09:02:18 +02:00
|
|
|
|
|
|
|
# capture multiple outputs
|
|
|
|
mshell --ssh ubuntu@test -- "cat file1.txt > /dev/fd/3; cat file2.txt > /dev/fd/4" 3> file1.txt 4> file2.txt
|
|
|
|
|
|
|
|
# execute a script, catpure stdout/stderr in fd-3 and fd-4
|
|
|
|
# useful if you need to see stdout for interacting with ssh (password or host auth)
|
|
|
|
mshell --ssh user@host -- "test.sh > /dev/fd/3 2> /dev/fd/4" 3> test.stdout 4> test.stderr
|
2022-06-23 21:48:45 +02:00
|
|
|
|
2022-06-26 10:53:07 +02:00
|
|
|
# run a script as root (via sudo), capture output
|
|
|
|
mshell --sudo-with-passfile pw.txt --ssh ubuntu@somehost -- "python3 /dev/fd/3 > /dev/fd/4" 3< myscript.py 4> script-output.txt < script-input.txt
|
|
|
|
|
2022-06-23 21:48:45 +02:00
|
|
|
mshell is licensed under the MPLv2
|
|
|
|
Please see https://github.com/scripthaus-dev/mshell for extended usage modes, source code, bugs, and feature requests
|
|
|
|
`
|
|
|
|
fmt.Printf("%s\n\n", strings.TrimSpace(usage))
|
|
|
|
}
|
|
|
|
|
2022-06-11 06:37:21 +02:00
|
|
|
func main() {
|
2022-06-23 21:48:45 +02:00
|
|
|
if len(os.Args) == 1 {
|
2022-06-24 09:02:18 +02:00
|
|
|
handleUsage()
|
2022-06-23 21:48:45 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
firstArg := os.Args[1]
|
|
|
|
if firstArg == "--help" {
|
2022-06-24 09:02:18 +02:00
|
|
|
handleUsage()
|
2022-06-23 21:48:45 +02:00
|
|
|
return
|
|
|
|
} else if firstArg == "--version" {
|
|
|
|
fmt.Printf("mshell v%s\n", MShellVersion)
|
|
|
|
return
|
|
|
|
} else if firstArg == "--remote" {
|
|
|
|
handleRemote()
|
|
|
|
return
|
|
|
|
} else if firstArg == "--server" {
|
|
|
|
handleServer()
|
|
|
|
return
|
|
|
|
} else {
|
2022-06-24 09:02:18 +02:00
|
|
|
rtnCode, err := handleClient()
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("[error] %v\n", err)
|
|
|
|
}
|
|
|
|
os.Exit(rtnCode)
|
2022-06-23 21:48:45 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-06-11 06:37:21 +02:00
|
|
|
if len(os.Args) >= 2 {
|
2022-06-27 21:03:47 +02:00
|
|
|
ck := base.CommandKey(os.Args[1])
|
|
|
|
if err := ck.Validate("mshell arg"); err != nil {
|
|
|
|
packet.SendErrorPacket(os.Stdout, err.Error())
|
2022-06-11 06:37:21 +02:00
|
|
|
return
|
|
|
|
}
|
2022-06-27 21:03:47 +02:00
|
|
|
doSingle(ck)
|
2022-06-17 21:27:29 +02:00
|
|
|
time.Sleep(100 * time.Millisecond)
|
2022-06-10 09:35:24 +02:00
|
|
|
return
|
2022-06-11 06:37:21 +02:00
|
|
|
} else {
|
|
|
|
doMain()
|
2022-06-10 09:35:24 +02:00
|
|
|
}
|
|
|
|
}
|