2022-11-29 03:03:02 +01:00
|
|
|
package cmdrunner
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"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/remote"
|
|
|
|
)
|
|
|
|
|
2022-12-28 08:12:27 +01:00
|
|
|
// PTERM=MxM,Mx25
|
|
|
|
// PTERM="Mx25!"
|
2022-11-29 03:03:02 +01:00
|
|
|
// PTERM=80x25,80x35
|
|
|
|
|
|
|
|
type PTermOptsType struct {
|
|
|
|
Rows string
|
|
|
|
RowsFlex bool
|
|
|
|
Cols string
|
|
|
|
ColsFlex bool
|
|
|
|
}
|
|
|
|
|
|
|
|
const PTermMax = "M"
|
|
|
|
|
|
|
|
func isDigits(s string) bool {
|
|
|
|
for _, ch := range s {
|
|
|
|
if ch < '0' || ch > '9' {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func atoiDefault(s string, def int) int {
|
|
|
|
ival, err := strconv.Atoi(s)
|
|
|
|
if err != nil {
|
|
|
|
return def
|
|
|
|
}
|
|
|
|
return ival
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseTermPart(part string, partType string) (string, bool, error) {
|
|
|
|
flex := true
|
|
|
|
if strings.HasSuffix(part, "!") {
|
|
|
|
part = part[:len(part)-1]
|
|
|
|
flex = false
|
|
|
|
}
|
|
|
|
if part == "" {
|
|
|
|
return PTermMax, flex, nil
|
|
|
|
}
|
|
|
|
if part == PTermMax {
|
|
|
|
return PTermMax, flex, nil
|
|
|
|
}
|
|
|
|
if !isDigits(part) {
|
|
|
|
return "", false, fmt.Errorf("invalid PTERM %s: must be '%s' or [number]", partType, PTermMax)
|
|
|
|
}
|
|
|
|
return part, flex, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseSingleTermStr(s string) (*PTermOptsType, error) {
|
|
|
|
s = strings.TrimSpace(s)
|
|
|
|
xIdx := strings.Index(s, "x")
|
|
|
|
if xIdx == -1 {
|
|
|
|
return nil, fmt.Errorf("invalid PTERM, must include 'x' to separate width and height (e.g. WxH)")
|
|
|
|
}
|
|
|
|
rowsPart := s[0:xIdx]
|
|
|
|
colsPart := s[xIdx+1:]
|
|
|
|
rows, rowsFlex, err := parseTermPart(rowsPart, "rows")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
cols, colsFlex, err := parseTermPart(colsPart, "cols")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &PTermOptsType{Rows: rows, RowsFlex: rowsFlex, Cols: cols, ColsFlex: colsFlex}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetUITermOpts(winSize *packet.WinSize, ptermStr string) (*packet.TermOpts, error) {
|
|
|
|
opts, err := parseSingleTermStr(ptermStr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
termOpts := &packet.TermOpts{Rows: shexec.DefaultTermRows, Cols: shexec.DefaultTermCols, Term: remote.DefaultTerm, MaxPtySize: shexec.DefaultMaxPtySize}
|
|
|
|
if winSize == nil {
|
|
|
|
winSize = &packet.WinSize{Rows: shexec.DefaultTermRows, Cols: shexec.DefaultTermCols}
|
|
|
|
}
|
|
|
|
if winSize.Rows == 0 {
|
|
|
|
winSize.Rows = shexec.DefaultTermRows
|
|
|
|
}
|
|
|
|
if winSize.Cols == 0 {
|
|
|
|
winSize.Cols = shexec.DefaultTermCols
|
|
|
|
}
|
|
|
|
if opts.Rows == PTermMax {
|
|
|
|
termOpts.Rows = winSize.Rows
|
|
|
|
} else {
|
|
|
|
termOpts.Rows = atoiDefault(opts.Rows, termOpts.Rows)
|
|
|
|
}
|
|
|
|
if opts.Cols == PTermMax {
|
|
|
|
termOpts.Cols = winSize.Cols
|
|
|
|
} else {
|
|
|
|
termOpts.Cols = atoiDefault(opts.Cols, termOpts.Cols)
|
|
|
|
}
|
|
|
|
termOpts.MaxPtySize = base.BoundInt64(termOpts.MaxPtySize, shexec.MinMaxPtySize, shexec.MaxMaxPtySize)
|
|
|
|
termOpts.Cols = base.BoundInt(termOpts.Cols, shexec.MinTermCols, shexec.MaxTermCols)
|
|
|
|
termOpts.Rows = base.BoundInt(termOpts.Rows, shexec.MinTermRows, shexec.MaxTermRows)
|
|
|
|
return termOpts, nil
|
|
|
|
}
|