2023-10-17 06:31:13 +02:00
|
|
|
// Copyright 2023, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2022-07-01 02:02:19 +02:00
|
|
|
package scbase
|
|
|
|
|
|
|
|
import (
|
2023-02-24 00:17:47 +01:00
|
|
|
"context"
|
2022-07-07 04:01:00 +02:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2022-11-02 05:42:56 +01:00
|
|
|
"io"
|
2022-07-07 04:01:00 +02:00
|
|
|
"io/fs"
|
2022-10-31 20:40:45 +01:00
|
|
|
"log"
|
2022-07-01 02:02:19 +02:00
|
|
|
"os"
|
2023-02-24 00:17:47 +01:00
|
|
|
"os/exec"
|
2023-10-05 18:42:48 +02:00
|
|
|
"os/user"
|
2022-07-01 02:02:19 +02:00
|
|
|
"path"
|
2023-02-24 00:17:47 +01:00
|
|
|
"regexp"
|
2022-12-29 08:09:37 +01:00
|
|
|
"runtime"
|
2022-09-21 02:37:49 +02:00
|
|
|
"strconv"
|
2023-02-24 00:17:47 +01:00
|
|
|
"strings"
|
2022-07-07 04:01:00 +02:00
|
|
|
"sync"
|
2023-02-24 00:17:47 +01:00
|
|
|
"time"
|
2022-07-08 01:29:14 +02:00
|
|
|
|
2023-07-31 02:16:43 +02:00
|
|
|
"github.com/google/uuid"
|
2023-10-16 23:02:22 +02:00
|
|
|
"github.com/wavetermdev/waveterm/waveshell/pkg/base"
|
2022-11-02 05:42:56 +01:00
|
|
|
"golang.org/x/mod/semver"
|
2022-07-08 01:29:14 +02:00
|
|
|
"golang.org/x/sys/unix"
|
2022-07-01 02:02:19 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const HomeVarName = "HOME"
|
2022-12-20 02:36:19 +01:00
|
|
|
const PromptHomeVarName = "PROMPT_HOME"
|
2022-12-28 22:56:19 +01:00
|
|
|
const PromptDevVarName = "PROMPT_DEV"
|
2022-07-07 04:01:00 +02:00
|
|
|
const SessionsDirBaseName = "sessions"
|
2023-03-21 03:20:57 +01:00
|
|
|
const ScreensDirBaseName = "screens"
|
2022-12-20 03:52:08 +01:00
|
|
|
const PromptLockFile = "prompt.lock"
|
2022-12-20 02:36:19 +01:00
|
|
|
const PromptDirName = "prompt"
|
2022-12-28 22:56:19 +01:00
|
|
|
const PromptDevDirName = "prompt-dev"
|
2022-12-20 02:36:19 +01:00
|
|
|
const PromptAppPathVarName = "PROMPT_APP_PATH"
|
2023-10-24 18:29:00 +02:00
|
|
|
const PromptVersion = "v0.5.0"
|
2022-12-20 03:52:08 +01:00
|
|
|
const PromptAuthKeyFileName = "prompt.authkey"
|
2023-09-07 06:50:38 +02:00
|
|
|
const MShellVersion = "v0.3.0"
|
2023-10-05 18:42:48 +02:00
|
|
|
const DefaultMacOSShell = "/bin/bash"
|
2022-07-07 04:01:00 +02:00
|
|
|
|
|
|
|
var SessionDirCache = make(map[string]string)
|
2023-03-21 03:20:57 +01:00
|
|
|
var ScreenDirCache = make(map[string]string)
|
2022-07-07 04:01:00 +02:00
|
|
|
var BaseLock = &sync.Mutex{}
|
2023-02-24 00:17:47 +01:00
|
|
|
var BuildTime = "-"
|
2022-07-01 02:02:19 +02:00
|
|
|
|
2022-12-29 02:47:12 +01:00
|
|
|
func IsDevMode() bool {
|
|
|
|
pdev := os.Getenv(PromptDevVarName)
|
|
|
|
return pdev != ""
|
|
|
|
}
|
|
|
|
|
2022-11-02 05:42:56 +01:00
|
|
|
// must match js
|
2022-12-20 02:36:19 +01:00
|
|
|
func GetPromptHomeDir() string {
|
|
|
|
scHome := os.Getenv(PromptHomeVarName)
|
2022-07-01 02:02:19 +02:00
|
|
|
if scHome == "" {
|
|
|
|
homeVar := os.Getenv(HomeVarName)
|
|
|
|
if homeVar == "" {
|
|
|
|
homeVar = "/"
|
|
|
|
}
|
2022-12-28 22:56:19 +01:00
|
|
|
pdev := os.Getenv(PromptDevVarName)
|
|
|
|
if pdev != "" {
|
|
|
|
scHome = path.Join(homeVar, PromptDevDirName)
|
|
|
|
} else {
|
|
|
|
scHome = path.Join(homeVar, PromptDirName)
|
|
|
|
}
|
|
|
|
|
2022-07-01 02:02:19 +02:00
|
|
|
}
|
|
|
|
return scHome
|
|
|
|
}
|
2022-07-07 04:01:00 +02:00
|
|
|
|
2022-12-29 08:09:37 +01:00
|
|
|
func MShellBinaryDir() string {
|
2022-12-20 02:36:19 +01:00
|
|
|
appPath := os.Getenv(PromptAppPathVarName)
|
2022-11-02 05:42:56 +01:00
|
|
|
if appPath == "" {
|
2022-12-29 08:09:37 +01:00
|
|
|
appPath = "."
|
2022-11-02 05:42:56 +01:00
|
|
|
}
|
2022-12-29 08:09:37 +01:00
|
|
|
if IsDevMode() {
|
2023-10-16 23:02:22 +02:00
|
|
|
return path.Join(appPath, "waveshell", "bin")
|
2022-12-29 08:09:37 +01:00
|
|
|
}
|
|
|
|
return path.Join(appPath, "bin", "mshell")
|
|
|
|
}
|
|
|
|
|
|
|
|
func MShellBinaryPath(version string, goos string, goarch string) (string, error) {
|
2022-11-02 05:42:56 +01:00
|
|
|
if !base.ValidGoArch(goos, goarch) {
|
2022-12-29 08:09:37 +01:00
|
|
|
return "", fmt.Errorf("invalid goos/goarch combination: %s/%s", goos, goarch)
|
2022-11-02 05:42:56 +01:00
|
|
|
}
|
2022-12-29 08:09:37 +01:00
|
|
|
binaryDir := MShellBinaryDir()
|
2022-11-02 05:42:56 +01:00
|
|
|
versionStr := semver.MajorMinor(version)
|
|
|
|
if versionStr == "" {
|
2022-12-29 08:09:37 +01:00
|
|
|
return "", fmt.Errorf("invalid mshell version: %q", version)
|
2022-11-02 05:42:56 +01:00
|
|
|
}
|
|
|
|
fileName := fmt.Sprintf("mshell-%s-%s.%s", versionStr, goos, goarch)
|
2022-12-29 08:09:37 +01:00
|
|
|
fullFileName := path.Join(binaryDir, fileName)
|
|
|
|
return fullFileName, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func LocalMShellBinaryPath() (string, error) {
|
|
|
|
return MShellBinaryPath(MShellVersion, runtime.GOOS, runtime.GOARCH)
|
|
|
|
}
|
|
|
|
|
|
|
|
func MShellBinaryReader(version string, goos string, goarch string) (io.ReadCloser, error) {
|
|
|
|
mshellPath, err := MShellBinaryPath(version, goos, goarch)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
fd, err := os.Open(mshellPath)
|
2022-11-02 05:42:56 +01:00
|
|
|
if err != nil {
|
2022-12-29 08:09:37 +01:00
|
|
|
return nil, fmt.Errorf("cannot open mshell binary %q: %v", mshellPath, err)
|
2022-11-02 05:42:56 +01:00
|
|
|
}
|
|
|
|
return fd, nil
|
|
|
|
}
|
|
|
|
|
2022-12-20 03:52:08 +01:00
|
|
|
func createPromptAuthKeyFile(fileName string) (string, error) {
|
|
|
|
fd, err := os.OpenFile(fileName, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0600)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
defer fd.Close()
|
|
|
|
keyStr := GenPromptUUID()
|
|
|
|
_, err = fd.Write([]byte(keyStr))
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return keyStr, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ReadPromptAuthKey() (string, error) {
|
|
|
|
homeDir := GetPromptHomeDir()
|
|
|
|
err := ensureDir(homeDir)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("cannot find/create PROMPT_HOME directory %q", homeDir)
|
|
|
|
}
|
|
|
|
fileName := path.Join(homeDir, PromptAuthKeyFileName)
|
|
|
|
fd, err := os.Open(fileName)
|
|
|
|
if err != nil && errors.Is(err, fs.ErrNotExist) {
|
|
|
|
return createPromptAuthKeyFile(fileName)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("error opening prompt authkey:%s: %v", fileName, err)
|
|
|
|
}
|
|
|
|
defer fd.Close()
|
|
|
|
buf, err := io.ReadAll(fd)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("error reading prompt authkey:%s: %v", fileName, err)
|
|
|
|
}
|
|
|
|
keyStr := string(buf)
|
|
|
|
_, err = uuid.Parse(keyStr)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("invalid authkey:%s format: %v", fileName, err)
|
|
|
|
}
|
|
|
|
return keyStr, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func AcquirePromptLock() (*os.File, error) {
|
2022-12-20 02:36:19 +01:00
|
|
|
homeDir := GetPromptHomeDir()
|
2022-10-31 20:40:45 +01:00
|
|
|
err := ensureDir(homeDir)
|
|
|
|
if err != nil {
|
2022-12-20 02:36:19 +01:00
|
|
|
return nil, fmt.Errorf("cannot find/create PROMPT_HOME directory %q", homeDir)
|
2022-10-31 20:40:45 +01:00
|
|
|
}
|
2022-12-20 03:52:08 +01:00
|
|
|
lockFileName := path.Join(homeDir, PromptLockFile)
|
2022-07-08 01:29:14 +02:00
|
|
|
fd, err := os.Create(lockFileName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
err = unix.Flock(int(fd.Fd()), unix.LOCK_EX|unix.LOCK_NB)
|
|
|
|
if err != nil {
|
|
|
|
fd.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return fd, nil
|
|
|
|
}
|
|
|
|
|
2023-03-21 03:20:57 +01:00
|
|
|
// deprecated (v0.1.8)
|
2022-07-07 04:01:00 +02:00
|
|
|
func EnsureSessionDir(sessionId string) (string, error) {
|
2022-08-24 22:21:54 +02:00
|
|
|
if sessionId == "" {
|
|
|
|
return "", fmt.Errorf("cannot get session dir for blank sessionid")
|
|
|
|
}
|
2022-07-07 04:01:00 +02:00
|
|
|
BaseLock.Lock()
|
|
|
|
sdir, ok := SessionDirCache[sessionId]
|
|
|
|
BaseLock.Unlock()
|
|
|
|
if ok {
|
|
|
|
return sdir, nil
|
|
|
|
}
|
2022-12-20 02:36:19 +01:00
|
|
|
scHome := GetPromptHomeDir()
|
2022-07-07 04:01:00 +02:00
|
|
|
sdir = path.Join(scHome, SessionsDirBaseName, sessionId)
|
|
|
|
err := ensureDir(sdir)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
BaseLock.Lock()
|
|
|
|
SessionDirCache[sessionId] = sdir
|
|
|
|
BaseLock.Unlock()
|
|
|
|
return sdir, nil
|
|
|
|
}
|
|
|
|
|
2023-03-21 03:20:57 +01:00
|
|
|
// deprecated (v0.1.8)
|
2022-12-27 03:49:45 +01:00
|
|
|
func GetSessionsDir() string {
|
|
|
|
promptHome := GetPromptHomeDir()
|
|
|
|
sdir := path.Join(promptHome, SessionsDirBaseName)
|
|
|
|
return sdir
|
|
|
|
}
|
|
|
|
|
2023-03-21 03:20:57 +01:00
|
|
|
func EnsureScreenDir(screenId string) (string, error) {
|
|
|
|
if screenId == "" {
|
|
|
|
return "", fmt.Errorf("cannot get screen dir for blank sessionid")
|
|
|
|
}
|
|
|
|
BaseLock.Lock()
|
|
|
|
sdir, ok := ScreenDirCache[screenId]
|
|
|
|
BaseLock.Unlock()
|
|
|
|
if ok {
|
|
|
|
return sdir, nil
|
|
|
|
}
|
|
|
|
scHome := GetPromptHomeDir()
|
|
|
|
sdir = path.Join(scHome, ScreensDirBaseName, screenId)
|
|
|
|
err := ensureDir(sdir)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
BaseLock.Lock()
|
|
|
|
ScreenDirCache[screenId] = sdir
|
|
|
|
BaseLock.Unlock()
|
|
|
|
return sdir, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetScreensDir() string {
|
|
|
|
promptHome := GetPromptHomeDir()
|
|
|
|
sdir := path.Join(promptHome, ScreensDirBaseName)
|
|
|
|
return sdir
|
|
|
|
}
|
|
|
|
|
2022-07-07 04:01:00 +02:00
|
|
|
func ensureDir(dirName string) error {
|
|
|
|
info, err := os.Stat(dirName)
|
|
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
|
|
err = os.MkdirAll(dirName, 0700)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-12-20 02:36:19 +01:00
|
|
|
log.Printf("[prompt] created directory %q\n", dirName)
|
2022-07-07 04:01:00 +02:00
|
|
|
info, err = os.Stat(dirName)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !info.IsDir() {
|
|
|
|
return fmt.Errorf("'%s' must be a directory", dirName)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-03-21 03:20:57 +01:00
|
|
|
// deprecated (v0.1.8)
|
|
|
|
func PtyOutFile_Sessions(sessionId string, cmdId string) (string, error) {
|
2022-07-07 04:01:00 +02:00
|
|
|
sdir, err := EnsureSessionDir(sessionId)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2022-08-24 22:21:54 +02:00
|
|
|
if sessionId == "" {
|
|
|
|
return "", fmt.Errorf("cannot get ptyout file for blank sessionid")
|
|
|
|
}
|
|
|
|
if cmdId == "" {
|
|
|
|
return "", fmt.Errorf("cannot get ptyout file for blank cmdid")
|
|
|
|
}
|
2022-08-19 22:23:00 +02:00
|
|
|
return fmt.Sprintf("%s/%s.ptyout.cf", sdir, cmdId), nil
|
2022-07-07 04:01:00 +02:00
|
|
|
}
|
|
|
|
|
2023-07-31 02:16:43 +02:00
|
|
|
func PtyOutFile(screenId string, lineId string) (string, error) {
|
2023-03-21 03:20:57 +01:00
|
|
|
sdir, err := EnsureScreenDir(screenId)
|
2022-07-07 04:01:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2023-03-21 03:20:57 +01:00
|
|
|
if screenId == "" {
|
|
|
|
return "", fmt.Errorf("cannot get ptyout file for blank screenid")
|
2022-08-24 22:21:54 +02:00
|
|
|
}
|
2023-07-31 02:16:43 +02:00
|
|
|
if lineId == "" {
|
|
|
|
return "", fmt.Errorf("cannot get ptyout file for blank lineid")
|
2022-08-24 22:21:54 +02:00
|
|
|
}
|
2023-07-31 02:16:43 +02:00
|
|
|
return fmt.Sprintf("%s/%s.ptyout.cf", sdir, lineId), nil
|
2022-07-08 06:39:25 +02:00
|
|
|
}
|
2022-09-21 02:37:49 +02:00
|
|
|
|
2022-12-20 03:52:08 +01:00
|
|
|
func GenPromptUUID() string {
|
2022-09-21 02:37:49 +02:00
|
|
|
for {
|
|
|
|
rtn := uuid.New().String()
|
|
|
|
_, err := strconv.Atoi(rtn[0:8])
|
|
|
|
if err == nil { // do not allow UUIDs where the initial 8 bytes parse to an integer
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return rtn
|
|
|
|
}
|
|
|
|
}
|
2022-09-21 21:39:55 +02:00
|
|
|
|
|
|
|
func NumFormatDec(num int64) string {
|
|
|
|
var signStr string
|
|
|
|
absNum := num
|
|
|
|
if absNum < 0 {
|
|
|
|
absNum = -absNum
|
|
|
|
signStr = "-"
|
|
|
|
}
|
|
|
|
if absNum < 1000 {
|
|
|
|
// raw num
|
|
|
|
return signStr + strconv.FormatInt(absNum, 10)
|
|
|
|
}
|
|
|
|
if absNum < 1000000 {
|
|
|
|
// k num
|
|
|
|
kVal := float64(absNum) / 1000
|
|
|
|
return signStr + strconv.FormatFloat(kVal, 'f', 2, 64) + "k"
|
|
|
|
}
|
|
|
|
if absNum < 1000000000 {
|
|
|
|
// M num
|
|
|
|
mVal := float64(absNum) / 1000000
|
|
|
|
return signStr + strconv.FormatFloat(mVal, 'f', 2, 64) + "m"
|
|
|
|
} else {
|
|
|
|
// G num
|
|
|
|
gVal := float64(absNum) / 1000000000
|
|
|
|
return signStr + strconv.FormatFloat(gVal, 'f', 2, 64) + "g"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func NumFormatB2(num int64) string {
|
|
|
|
var signStr string
|
|
|
|
absNum := num
|
|
|
|
if absNum < 0 {
|
|
|
|
absNum = -absNum
|
|
|
|
signStr = "-"
|
|
|
|
}
|
|
|
|
if absNum < 1024 {
|
|
|
|
// raw num
|
|
|
|
return signStr + strconv.FormatInt(absNum, 10)
|
|
|
|
}
|
|
|
|
if absNum < 1000000 {
|
|
|
|
// k num
|
|
|
|
if absNum%1024 == 0 {
|
|
|
|
return signStr + strconv.FormatInt(absNum/1024, 10) + "K"
|
|
|
|
}
|
|
|
|
kVal := float64(absNum) / 1024
|
|
|
|
return signStr + strconv.FormatFloat(kVal, 'f', 2, 64) + "K"
|
|
|
|
}
|
|
|
|
if absNum < 1000000000 {
|
|
|
|
// M num
|
|
|
|
if absNum%(1024*1024) == 0 {
|
|
|
|
return signStr + strconv.FormatInt(absNum/(1024*1024), 10) + "M"
|
|
|
|
}
|
|
|
|
mVal := float64(absNum) / (1024 * 1024)
|
|
|
|
return signStr + strconv.FormatFloat(mVal, 'f', 2, 64) + "M"
|
|
|
|
} else {
|
|
|
|
// G num
|
|
|
|
if absNum%(1024*1024*1024) == 0 {
|
|
|
|
return signStr + strconv.FormatInt(absNum/(1024*1024*1024), 10) + "G"
|
|
|
|
}
|
|
|
|
gVal := float64(absNum) / (1024 * 1024 * 1024)
|
|
|
|
return signStr + strconv.FormatFloat(gVal, 'f', 2, 64) + "G"
|
|
|
|
}
|
|
|
|
}
|
2023-01-17 08:36:52 +01:00
|
|
|
|
|
|
|
func ClientArch() string {
|
|
|
|
return fmt.Sprintf("%s/%s", runtime.GOOS, runtime.GOARCH)
|
|
|
|
}
|
2023-02-24 00:17:47 +01:00
|
|
|
|
|
|
|
var releaseRegex = regexp.MustCompile(`^\d+\.\d+\.\d+$`)
|
|
|
|
var osReleaseOnce = &sync.Once{}
|
|
|
|
var osRelease string
|
|
|
|
|
|
|
|
func macOSRelease() string {
|
|
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), 2*time.Second)
|
|
|
|
defer cancelFn()
|
|
|
|
out, err := exec.CommandContext(ctx, "uname", "-r").CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error executing uname -r: %v\n", err)
|
|
|
|
return "-"
|
|
|
|
}
|
|
|
|
releaseStr := strings.TrimSpace(string(out))
|
|
|
|
if !releaseRegex.MatchString(releaseStr) {
|
|
|
|
log.Printf("invalid uname -r output: [%s]\n", releaseStr)
|
|
|
|
return "-"
|
|
|
|
}
|
|
|
|
return releaseStr
|
|
|
|
}
|
|
|
|
|
|
|
|
func MacOSRelease() string {
|
|
|
|
osReleaseOnce.Do(func() {
|
|
|
|
osRelease = macOSRelease()
|
|
|
|
})
|
|
|
|
return osRelease
|
|
|
|
}
|
2023-10-05 18:42:48 +02:00
|
|
|
|
|
|
|
var userShellRegexp = regexp.MustCompile(`^UserShell: (.*)$`)
|
|
|
|
|
|
|
|
// dscl . -read /User/[username] UserShell
|
|
|
|
// defaults to /bin/bash
|
|
|
|
func MacUserShell() string {
|
|
|
|
osUser, err := user.Current()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error getting current user: %v\n", err)
|
|
|
|
return DefaultMacOSShell
|
|
|
|
}
|
|
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), 2*time.Second)
|
|
|
|
defer cancelFn()
|
|
|
|
userStr := "/Users/" + osUser.Name
|
|
|
|
out, err := exec.CommandContext(ctx, "dscl", ".", "-read", userStr, "UserShell").CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error executing macos user shell lookup: %v %q\n", err, string(out))
|
|
|
|
return DefaultMacOSShell
|
|
|
|
}
|
|
|
|
outStr := strings.TrimSpace(string(out))
|
|
|
|
m := userShellRegexp.FindStringSubmatch(outStr)
|
|
|
|
if m == nil {
|
|
|
|
log.Printf("error in format of dscl output: %q\n", outStr)
|
|
|
|
return DefaultMacOSShell
|
|
|
|
}
|
|
|
|
return m[1]
|
|
|
|
}
|