2024-10-24 07:43:17 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-08-16 06:32:08 +02:00
|
|
|
package remote
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"html/template"
|
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"os"
|
2024-09-06 02:02:44 +02:00
|
|
|
"os/user"
|
2024-08-16 06:32:08 +02:00
|
|
|
"path/filepath"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"golang.org/x/crypto/ssh"
|
|
|
|
)
|
|
|
|
|
2024-10-24 19:58:31 +02:00
|
|
|
var userHostRe = regexp.MustCompile(`^([a-zA-Z0-9][a-zA-Z0-9._@\\-]*@)?([a-zA-Z0-9][a-zA-Z0-9.-]*)(?::([0-9]+))?$`)
|
2024-08-16 06:32:08 +02:00
|
|
|
|
|
|
|
func ParseOpts(input string) (*SSHOpts, error) {
|
|
|
|
m := userHostRe.FindStringSubmatch(input)
|
|
|
|
if m == nil {
|
|
|
|
return nil, fmt.Errorf("invalid format of user@host argument")
|
|
|
|
}
|
|
|
|
remoteUser, remoteHost, remotePortStr := m[1], m[2], m[3]
|
|
|
|
remoteUser = strings.Trim(remoteUser, "@")
|
|
|
|
var remotePort int
|
|
|
|
if remotePortStr != "" {
|
|
|
|
var err error
|
|
|
|
remotePort, err = strconv.Atoi(remotePortStr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("invalid port specified on user@host argument")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &SSHOpts{SSHHost: remoteHost, SSHUser: remoteUser, SSHPort: remotePort}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func DetectShell(client *ssh.Client) (string, error) {
|
2024-08-19 06:26:44 +02:00
|
|
|
wshPath := GetWshPath(client)
|
2024-08-16 06:32:08 +02:00
|
|
|
|
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("shell detecting using command: %s shell", wshPath)
|
|
|
|
out, err := session.Output(wshPath + " shell")
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("unable to determine shell. defaulting to /bin/bash: %s", err)
|
|
|
|
return "/bin/bash", nil
|
|
|
|
}
|
|
|
|
log.Printf("detecting shell: %s", out)
|
|
|
|
|
|
|
|
return fmt.Sprintf(`"%s"`, strings.TrimSpace(string(out))), nil
|
|
|
|
}
|
|
|
|
|
2024-08-19 06:26:44 +02:00
|
|
|
func GetWshVersion(client *ssh.Client) (string, error) {
|
|
|
|
wshPath := GetWshPath(client)
|
2024-08-16 06:32:08 +02:00
|
|
|
|
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, err := session.Output(wshPath + " version")
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.TrimSpace(string(out)), nil
|
|
|
|
}
|
|
|
|
|
2024-08-19 06:26:44 +02:00
|
|
|
func GetWshPath(client *ssh.Client) string {
|
2024-08-20 21:42:43 +02:00
|
|
|
defaultPath := "~/.waveterm/bin/wsh"
|
2024-08-16 06:32:08 +02:00
|
|
|
|
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("unable to detect client's wsh path. using default. error: %v", err)
|
|
|
|
return defaultPath
|
|
|
|
}
|
|
|
|
|
|
|
|
out, whichErr := session.Output("which wsh")
|
|
|
|
if whichErr == nil {
|
|
|
|
return strings.TrimSpace(string(out))
|
|
|
|
}
|
|
|
|
|
|
|
|
session, err = client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("unable to detect client's wsh path. using default. error: %v", err)
|
|
|
|
return defaultPath
|
|
|
|
}
|
|
|
|
|
|
|
|
out, whereErr := session.Output("where.exe wsh")
|
|
|
|
if whereErr == nil {
|
|
|
|
return strings.TrimSpace(string(out))
|
|
|
|
}
|
|
|
|
|
2024-08-20 21:42:43 +02:00
|
|
|
// check cmd on windows since it requires an absolute path with backslashes
|
|
|
|
session, err = client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("unable to detect client's wsh path. using default. error: %v", err)
|
|
|
|
return defaultPath
|
|
|
|
}
|
|
|
|
|
|
|
|
out, cmdErr := session.Output("(dir 2>&1 *``|echo %userprofile%\\.waveterm%\\.waveterm\\bin\\wsh.exe);&<# rem #>echo none") //todo
|
|
|
|
if cmdErr == nil && strings.TrimSpace(string(out)) != "none" {
|
|
|
|
return strings.TrimSpace(string(out))
|
|
|
|
}
|
|
|
|
|
2024-08-16 06:32:08 +02:00
|
|
|
// no custom install, use default path
|
|
|
|
return defaultPath
|
|
|
|
}
|
|
|
|
|
|
|
|
func hasBashInstalled(client *ssh.Client) (bool, error) {
|
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
// this is a true error that should stop further progress
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, whichErr := session.Output("which bash")
|
|
|
|
if whichErr == nil && len(out) != 0 {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
session, err = client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
// this is a true error that should stop further progress
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, whereErr := session.Output("where.exe bash")
|
|
|
|
if whereErr == nil && len(out) != 0 {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// note: we could also check in /bin/bash explicitly
|
|
|
|
// just in case that wasn't added to the path. but if
|
|
|
|
// that's true, we will most likely have worse
|
|
|
|
// problems going forward
|
|
|
|
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
2024-08-19 06:26:44 +02:00
|
|
|
func GetClientOs(client *ssh.Client) (string, error) {
|
2024-08-16 06:32:08 +02:00
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, unixErr := session.Output("uname -s")
|
|
|
|
if unixErr == nil {
|
|
|
|
formatted := strings.ToLower(string(out))
|
|
|
|
formatted = strings.TrimSpace(formatted)
|
|
|
|
return formatted, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
session, err = client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, cmdErr := session.Output("echo %OS%")
|
|
|
|
if cmdErr == nil {
|
|
|
|
formatted := strings.ToLower(string(out))
|
|
|
|
formatted = strings.TrimSpace(formatted)
|
|
|
|
return strings.Split(formatted, "_")[0], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
session, err = client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, psErr := session.Output("echo $env:OS")
|
|
|
|
if psErr == nil {
|
|
|
|
formatted := strings.ToLower(string(out))
|
|
|
|
formatted = strings.TrimSpace(formatted)
|
|
|
|
return strings.Split(formatted, "_")[0], nil
|
|
|
|
}
|
|
|
|
return "", fmt.Errorf("unable to determine os: {unix: %s, cmd: %s, powershell: %s}", unixErr, cmdErr, psErr)
|
|
|
|
}
|
|
|
|
|
2024-08-19 06:26:44 +02:00
|
|
|
func GetClientArch(client *ssh.Client) (string, error) {
|
2024-08-16 06:32:08 +02:00
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, unixErr := session.Output("uname -m")
|
|
|
|
if unixErr == nil {
|
|
|
|
formatted := strings.ToLower(string(out))
|
|
|
|
formatted = strings.TrimSpace(formatted)
|
|
|
|
if formatted == "x86_64" {
|
2024-09-04 20:23:39 +02:00
|
|
|
return "x64", nil
|
2024-08-16 06:32:08 +02:00
|
|
|
}
|
|
|
|
return formatted, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
session, err = client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, cmdErr := session.Output("echo %PROCESSOR_ARCHITECTURE%")
|
|
|
|
if cmdErr == nil {
|
|
|
|
formatted := strings.ToLower(string(out))
|
|
|
|
return strings.TrimSpace(formatted), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
session, err = client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out, psErr := session.Output("echo $env:PROCESSOR_ARCHITECTURE")
|
|
|
|
if psErr == nil {
|
|
|
|
formatted := strings.ToLower(string(out))
|
|
|
|
return strings.TrimSpace(formatted), nil
|
|
|
|
}
|
|
|
|
return "", fmt.Errorf("unable to determine architecture: {unix: %s, cmd: %s, powershell: %s}", unixErr, cmdErr, psErr)
|
|
|
|
}
|
|
|
|
|
|
|
|
var installTemplateRawBash = `bash -c ' \
|
|
|
|
mkdir -p {{.installDir}}; \
|
|
|
|
cat > {{.tempPath}}; \
|
|
|
|
mv {{.tempPath}} {{.installPath}}; \
|
|
|
|
chmod a+x {{.installPath}};' \
|
|
|
|
`
|
|
|
|
|
|
|
|
var installTemplateRawDefault = ` \
|
|
|
|
mkdir -p {{.installDir}}; \
|
|
|
|
cat > {{.tempPath}}; \
|
|
|
|
mv {{.tempPath}} {{.installPath}}; \
|
|
|
|
chmod a+x {{.installPath}}; \
|
|
|
|
`
|
|
|
|
|
2024-08-19 06:26:44 +02:00
|
|
|
func CpHostToRemote(client *ssh.Client, sourcePath string, destPath string) error {
|
2024-08-16 06:32:08 +02:00
|
|
|
// warning: does not work on windows remote yet
|
|
|
|
bashInstalled, err := hasBashInstalled(client)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var selectedTemplateRaw string
|
|
|
|
if bashInstalled {
|
|
|
|
selectedTemplateRaw = installTemplateRawBash
|
|
|
|
} else {
|
|
|
|
log.Printf("bash is not installed on remote. attempting with default shell")
|
|
|
|
selectedTemplateRaw = installTemplateRawDefault
|
|
|
|
}
|
|
|
|
|
2024-09-28 01:41:53 +02:00
|
|
|
// I need to use toSlash here to force unix keybindings
|
|
|
|
// this means we can't guarantee it will work on a remote windows machine
|
2024-08-16 06:32:08 +02:00
|
|
|
var installWords = map[string]string{
|
2024-09-28 01:41:53 +02:00
|
|
|
"installDir": filepath.ToSlash(filepath.Dir(destPath)),
|
2024-08-16 06:32:08 +02:00
|
|
|
"tempPath": destPath + ".temp",
|
|
|
|
"installPath": destPath,
|
|
|
|
}
|
|
|
|
|
|
|
|
installCmd := &bytes.Buffer{}
|
|
|
|
installTemplate := template.Must(template.New("").Parse(selectedTemplateRaw))
|
|
|
|
installTemplate.Execute(installCmd, installWords)
|
|
|
|
|
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
installStdin, err := session.StdinPipe()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = session.Start(installCmd.String())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
input, err := os.Open(sourcePath)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("cannot open local file %s to send to host: %v", sourcePath, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
io.Copy(installStdin, input)
|
|
|
|
session.Close() // this allows the command to complete for reasons i don't fully understand
|
|
|
|
}()
|
|
|
|
|
|
|
|
return session.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
func InstallClientRcFiles(client *ssh.Client) error {
|
2024-08-19 06:26:44 +02:00
|
|
|
path := GetWshPath(client)
|
2024-08-20 21:42:43 +02:00
|
|
|
log.Printf("path to wsh searched is: %s", path)
|
2024-08-16 06:32:08 +02:00
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
// this is a true error that should stop further progress
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = session.Output(path + " rcfiles")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetHomeDir(client *ssh.Client) string {
|
|
|
|
session, err := client.NewSession()
|
|
|
|
if err != nil {
|
|
|
|
return "~"
|
|
|
|
}
|
|
|
|
|
2024-09-04 11:13:00 +02:00
|
|
|
out, err := session.Output(`echo "$HOME"`)
|
|
|
|
if err == nil {
|
|
|
|
return strings.TrimSpace(string(out))
|
|
|
|
}
|
|
|
|
|
|
|
|
session, err = client.NewSession()
|
2024-08-16 06:32:08 +02:00
|
|
|
if err != nil {
|
|
|
|
return "~"
|
|
|
|
}
|
2024-09-04 11:13:00 +02:00
|
|
|
out, err = session.Output(`echo %userprofile%`)
|
|
|
|
if err == nil {
|
|
|
|
return strings.TrimSpace(string(out))
|
|
|
|
}
|
|
|
|
|
|
|
|
return "~"
|
|
|
|
}
|
2024-08-16 06:32:08 +02:00
|
|
|
|
2024-09-04 11:13:00 +02:00
|
|
|
func IsPowershell(shellPath string) bool {
|
|
|
|
// get the base path, and then check contains
|
|
|
|
shellBase := filepath.Base(shellPath)
|
|
|
|
return strings.Contains(shellBase, "powershell") || strings.Contains(shellBase, "pwsh")
|
2024-08-16 06:32:08 +02:00
|
|
|
}
|
2024-09-06 02:02:44 +02:00
|
|
|
|
|
|
|
func NormalizeConfigPattern(pattern string) string {
|
2024-10-28 04:35:19 +01:00
|
|
|
userName, err := WaveSshConfigUserSettings().GetStrict(pattern, "User")
|
2024-09-06 02:02:44 +02:00
|
|
|
if err != nil {
|
|
|
|
localUser, err := user.Current()
|
|
|
|
if err == nil {
|
|
|
|
userName = localUser.Username
|
|
|
|
}
|
|
|
|
}
|
2024-10-28 04:35:19 +01:00
|
|
|
port, err := WaveSshConfigUserSettings().GetStrict(pattern, "Port")
|
2024-09-06 02:02:44 +02:00
|
|
|
if err != nil {
|
|
|
|
port = "22"
|
|
|
|
}
|
|
|
|
if userName != "" {
|
|
|
|
userName += "@"
|
|
|
|
}
|
|
|
|
if port == "22" {
|
|
|
|
port = ""
|
|
|
|
} else {
|
|
|
|
port = ":" + port
|
|
|
|
}
|
2024-10-09 20:08:07 +02:00
|
|
|
return fmt.Sprintf("%s%s%s", userName, pattern, port)
|
2024-09-06 02:02:44 +02:00
|
|
|
}
|