2022-07-01 23:07:13 +02:00
|
|
|
package sstore
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"database/sql"
|
|
|
|
"fmt"
|
2022-07-13 06:51:17 +02:00
|
|
|
"strings"
|
2022-07-01 23:07:13 +02:00
|
|
|
|
|
|
|
"github.com/google/uuid"
|
2022-07-08 01:29:14 +02:00
|
|
|
"github.com/scripthaus-dev/mshell/pkg/packet"
|
2022-07-01 23:07:13 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func NumSessions(ctx context.Context) (int, error) {
|
2022-07-13 01:10:46 +02:00
|
|
|
db, err := GetDB(ctx)
|
2022-07-01 23:07:13 +02:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
query := "SELECT count(*) FROM session"
|
|
|
|
var count int
|
|
|
|
err = db.GetContext(ctx, &count, query)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return count, nil
|
|
|
|
}
|
|
|
|
|
2022-07-02 02:38:36 +02:00
|
|
|
func GetAllRemotes(ctx context.Context) ([]*RemoteType, error) {
|
2022-07-07 22:26:46 +02:00
|
|
|
var rtn []*RemoteType
|
|
|
|
err := WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `SELECT * FROM remote`
|
|
|
|
marr := tx.SelectMaps(query)
|
|
|
|
for _, m := range marr {
|
|
|
|
rtn = append(rtn, RemoteFromMap(m))
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
2022-07-02 02:38:36 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-07 22:26:46 +02:00
|
|
|
return rtn, nil
|
2022-07-02 02:38:36 +02:00
|
|
|
}
|
|
|
|
|
2022-07-01 23:07:13 +02:00
|
|
|
func GetRemoteByName(ctx context.Context, remoteName string) (*RemoteType, error) {
|
2022-07-07 22:26:46 +02:00
|
|
|
var remote *RemoteType
|
|
|
|
err := WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `SELECT * FROM remote WHERE remotename = ?`
|
|
|
|
m := tx.GetMap(query, remoteName)
|
|
|
|
remote = RemoteFromMap(m)
|
|
|
|
return nil
|
|
|
|
})
|
2022-07-01 23:07:13 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-07 22:26:46 +02:00
|
|
|
return remote, nil
|
2022-07-01 23:07:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func GetRemoteById(ctx context.Context, remoteId string) (*RemoteType, error) {
|
2022-07-07 22:26:46 +02:00
|
|
|
var remote *RemoteType
|
|
|
|
err := WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `SELECT * FROM remote WHERE remoteid = ?`
|
|
|
|
m := tx.GetMap(query, remoteId)
|
|
|
|
remote = RemoteFromMap(m)
|
|
|
|
return nil
|
|
|
|
})
|
2022-07-01 23:07:13 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-07 22:26:46 +02:00
|
|
|
return remote, nil
|
2022-07-01 23:07:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func InsertRemote(ctx context.Context, remote *RemoteType) error {
|
|
|
|
if remote == nil {
|
|
|
|
return fmt.Errorf("cannot insert nil remote")
|
|
|
|
}
|
2022-07-13 01:10:46 +02:00
|
|
|
db, err := GetDB(ctx)
|
2022-07-01 23:07:13 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-07-07 22:26:46 +02:00
|
|
|
query := `INSERT INTO remote ( remoteid, remotetype, remotename, autoconnect, initpk, sshopts, lastconnectts) VALUES
|
|
|
|
(:remoteid,:remotetype,:remotename,:autoconnect,:initpk,:sshopts,:lastconnectts)`
|
|
|
|
_, err = db.NamedExec(query, remote.ToMap())
|
2022-07-01 23:07:13 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-07-08 22:23:45 +02:00
|
|
|
func GetAllSessions(ctx context.Context) ([]*SessionType, error) {
|
|
|
|
var rtn []*SessionType
|
2022-07-12 22:50:44 +02:00
|
|
|
err := WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `SELECT * FROM session`
|
|
|
|
tx.SelectWrap(&rtn, query)
|
|
|
|
var windows []*WindowType
|
|
|
|
query = `SELECT * FROM window`
|
|
|
|
tx.SelectWrap(&windows, query)
|
|
|
|
winMap := make(map[string][]*WindowType)
|
|
|
|
for _, win := range windows {
|
|
|
|
winArr := winMap[win.SessionId]
|
|
|
|
winArr = append(winArr, win)
|
|
|
|
winMap[win.SessionId] = winArr
|
|
|
|
}
|
|
|
|
for _, session := range rtn {
|
|
|
|
session.Windows = winMap[session.SessionId]
|
|
|
|
}
|
2022-07-13 06:51:17 +02:00
|
|
|
var screens []*ScreenType
|
|
|
|
query = `SELECT * FROM screen ORDER BY screenidx`
|
|
|
|
tx.SelectWrap(&screens, query)
|
|
|
|
screenMap := make(map[string][]*ScreenType)
|
|
|
|
for _, screen := range screens {
|
|
|
|
screenArr := screenMap[screen.SessionId]
|
|
|
|
screenArr = append(screenArr, screen)
|
|
|
|
screenMap[screen.SessionId] = screenArr
|
|
|
|
}
|
|
|
|
for _, session := range rtn {
|
|
|
|
session.Screens = screenMap[session.SessionId]
|
|
|
|
}
|
|
|
|
var sws []*ScreenWindowType
|
|
|
|
query = `SELECT * FROM screen_window`
|
|
|
|
tx.SelectWrap(&sws, query)
|
|
|
|
screenIdMap := make(map[string]*ScreenType)
|
|
|
|
for _, screen := range screens {
|
|
|
|
screenIdMap[screen.SessionId+screen.ScreenId] = screen
|
|
|
|
}
|
|
|
|
for _, sw := range sws {
|
|
|
|
screen := screenIdMap[sw.SessionId+sw.ScreenId]
|
|
|
|
if screen == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
screen.Windows = append(screen.Windows, sw)
|
|
|
|
}
|
2022-07-12 22:50:44 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return rtn, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetWindowById(ctx context.Context, sessionId string, windowId string) (*WindowType, error) {
|
|
|
|
var rtnWindow *WindowType
|
|
|
|
err := WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
var window WindowType
|
|
|
|
query := `SELECT * FROM window WHERE sessionid = ? AND windowid = ?`
|
|
|
|
found := tx.GetWrap(&window, query, sessionId, windowId)
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
rtnWindow = &window
|
|
|
|
query = `SELECT * FROM line WHERE sessionid = ? AND windowid = ?`
|
|
|
|
tx.SelectWrap(&window.Lines, query, sessionId, windowId)
|
|
|
|
query = `SELECT * FROM cmd WHERE cmdid IN (SELECT cmdid FROM line WHERE sessionid = ? AND windowid = ?)`
|
|
|
|
cmdMaps := tx.SelectMaps(query, sessionId, windowId)
|
|
|
|
for _, m := range cmdMaps {
|
|
|
|
window.Cmds = append(window.Cmds, CmdFromMap(m))
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return rtnWindow, err
|
2022-07-08 22:23:45 +02:00
|
|
|
}
|
|
|
|
|
2022-07-01 23:07:13 +02:00
|
|
|
func GetSessionById(ctx context.Context, id string) (*SessionType, error) {
|
2022-07-01 23:45:33 +02:00
|
|
|
var rtnSession *SessionType
|
|
|
|
err := WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
var session SessionType
|
2022-07-05 07:18:01 +02:00
|
|
|
query := `SELECT * FROM session WHERE sessionid = ?`
|
|
|
|
found := tx.GetWrap(&session, query, id)
|
2022-07-01 23:45:33 +02:00
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
rtnSession = &session
|
2022-07-12 23:27:16 +02:00
|
|
|
query = `SELECT * FROM window WHERE sessionid = ?`
|
2022-07-01 23:45:33 +02:00
|
|
|
tx.SelectWrap(&session.Windows, query, session.SessionId)
|
2022-07-06 01:54:49 +02:00
|
|
|
query = `SELECT * FROM remote_instance WHERE sessionid = ?`
|
2022-07-05 07:18:01 +02:00
|
|
|
tx.SelectWrap(&session.Remotes, query, session.SessionId)
|
2022-07-07 09:10:37 +02:00
|
|
|
query = `SELECT * FROM cmd WHERE sessionid = ?`
|
|
|
|
marr := tx.SelectMaps(query, session.SessionId)
|
|
|
|
for _, m := range marr {
|
|
|
|
session.Cmds = append(session.Cmds, CmdFromMap(m))
|
|
|
|
}
|
2022-07-01 23:45:33 +02:00
|
|
|
return nil
|
|
|
|
})
|
2022-07-01 23:07:13 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-01 23:45:33 +02:00
|
|
|
return rtnSession, nil
|
2022-07-01 23:07:13 +02:00
|
|
|
}
|
|
|
|
|
2022-07-05 07:18:01 +02:00
|
|
|
func GetSessionByName(ctx context.Context, name string) (*SessionType, error) {
|
2022-07-13 01:10:46 +02:00
|
|
|
db, err := GetDB(ctx)
|
2022-07-05 07:18:01 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
var sessionId string
|
|
|
|
query := `SELECT sessionid FROM session WHERE name = ?`
|
|
|
|
err = db.GetContext(ctx, &sessionId, query, name)
|
|
|
|
if err != nil {
|
|
|
|
if err == sql.ErrNoRows {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return GetSessionById(ctx, sessionId)
|
|
|
|
}
|
|
|
|
|
2022-07-08 22:23:45 +02:00
|
|
|
// also creates default window, returns sessionId
|
|
|
|
// if sessionName == "", it will be generated
|
|
|
|
func InsertSessionWithName(ctx context.Context, sessionName string) (string, error) {
|
|
|
|
newSessionId := uuid.New().String()
|
|
|
|
txErr := WithTx(ctx, func(tx *TxWrap) error {
|
2022-07-13 06:51:17 +02:00
|
|
|
names := tx.SelectStrings(`SELECT name FROM session`)
|
|
|
|
sessionName = fmtUniqueName(sessionName, "session-%d", len(names)+1, names)
|
|
|
|
maxSessionIdx := tx.GetInt(`SELECT COALESCE(max(sessionidx), 0) FROM session`)
|
|
|
|
query := `INSERT INTO session (sessionid, name, activescreenid, sessionidx, notifynum) VALUES (?, ?, '', ?, ?)`
|
|
|
|
tx.ExecWrap(query, newSessionId, sessionName, maxSessionIdx+1, 0)
|
|
|
|
screenId, err := InsertScreen(tx.Context(), newSessionId, "")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2022-07-08 22:23:45 +02:00
|
|
|
}
|
2022-07-13 06:51:17 +02:00
|
|
|
query = `UPDATE session SET activescreenid = ? WHERE sessionid = ?`
|
|
|
|
tx.ExecWrap(query, screenId, newSessionId)
|
2022-07-01 23:07:13 +02:00
|
|
|
return nil
|
|
|
|
})
|
2022-07-08 22:23:45 +02:00
|
|
|
return newSessionId, txErr
|
|
|
|
}
|
|
|
|
|
|
|
|
func containsStr(strs []string, testStr string) bool {
|
|
|
|
for _, s := range strs {
|
|
|
|
if s == testStr {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2022-07-13 06:51:17 +02:00
|
|
|
func fmtUniqueName(name string, defaultFmtStr string, startIdx int, strs []string) string {
|
|
|
|
var fmtStr string
|
|
|
|
if name != "" {
|
|
|
|
if !containsStr(strs, name) {
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
fmtStr = name + "-%d"
|
|
|
|
startIdx = 2
|
|
|
|
} else {
|
|
|
|
fmtStr = defaultFmtStr
|
|
|
|
}
|
|
|
|
if strings.Index(fmtStr, "%d") == -1 {
|
|
|
|
panic("invalid fmtStr: " + fmtStr)
|
|
|
|
}
|
|
|
|
for {
|
|
|
|
testName := fmt.Sprintf(fmtStr, startIdx)
|
|
|
|
if containsStr(strs, testName) {
|
|
|
|
startIdx++
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return testName
|
|
|
|
}
|
2022-07-13 01:10:46 +02:00
|
|
|
}
|
|
|
|
|
2022-07-13 06:51:17 +02:00
|
|
|
func InsertScreen(ctx context.Context, sessionId string, screenName string) (string, error) {
|
|
|
|
var newScreenId string
|
2022-07-08 22:23:45 +02:00
|
|
|
txErr := WithTx(ctx, func(tx *TxWrap) error {
|
2022-07-13 06:51:17 +02:00
|
|
|
query := `SELECT sessionid FROM session WHERE sessionid = ?`
|
|
|
|
if !tx.Exists(query, sessionId) {
|
|
|
|
return fmt.Errorf("cannot create screen, no session found")
|
2022-07-08 22:23:45 +02:00
|
|
|
}
|
2022-07-13 06:51:17 +02:00
|
|
|
newWindowId := txCreateWindow(tx, sessionId)
|
|
|
|
maxScreenIdx := tx.GetInt(`SELECT COALESCE(max(screenidx), 0) FROM screen WHERE sessionid = ?`, sessionId)
|
|
|
|
screenNames := tx.SelectStrings(`SELECT name FROM screen WHERE sessionid = ?`, sessionId)
|
|
|
|
screenName = fmtUniqueName(screenName, "s%d", maxScreenIdx+1, screenNames)
|
|
|
|
newScreenId = uuid.New().String()
|
|
|
|
query = `INSERT INTO screen (sessionid, screenid, name, activewindowid, screenidx, screenopts) VALUES (?, ?, ?, ?, ?, ?)`
|
|
|
|
tx.ExecWrap(query, sessionId, newScreenId, screenName, newWindowId, maxScreenIdx+1, ScreenOptsType{})
|
|
|
|
layout := LayoutType{Type: LayoutFull}
|
|
|
|
query = `INSERT INTO screen_window (sessionid, screenid, windowid, name, layout) VALUES (?, ?, ?, ?, ?)`
|
|
|
|
tx.ExecWrap(query, sessionId, newScreenId, newWindowId, DefaultScreenWindowName, layout)
|
2022-07-08 22:23:45 +02:00
|
|
|
return nil
|
|
|
|
})
|
2022-07-13 06:51:17 +02:00
|
|
|
return newScreenId, txErr
|
2022-07-01 23:07:13 +02:00
|
|
|
}
|
2022-07-02 22:31:56 +02:00
|
|
|
|
2022-07-13 06:51:17 +02:00
|
|
|
func txCreateWindow(tx *TxWrap, sessionId string) string {
|
|
|
|
windowId := uuid.New().String()
|
|
|
|
query := `INSERT INTO window (sessionid, windowid, curremote, winopts) VALUES (?, ?, ?, ?)`
|
|
|
|
tx.ExecWrap(query, sessionId, windowId, LocalRemoteName, WindowOptsType{})
|
|
|
|
return windowId
|
2022-07-12 22:50:44 +02:00
|
|
|
}
|
|
|
|
|
2022-07-08 06:39:25 +02:00
|
|
|
func InsertLine(ctx context.Context, line *LineType, cmd *CmdType) error {
|
2022-07-02 22:31:56 +02:00
|
|
|
if line == nil {
|
|
|
|
return fmt.Errorf("line cannot be nil")
|
|
|
|
}
|
|
|
|
if line.LineId != 0 {
|
|
|
|
return fmt.Errorf("new line cannot have LineId set")
|
|
|
|
}
|
|
|
|
return WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
var windowId string
|
|
|
|
query := `SELECT windowid FROM window WHERE sessionid = ? AND windowid = ?`
|
|
|
|
hasWindow := tx.GetWrap(&windowId, query, line.SessionId, line.WindowId)
|
|
|
|
if !hasWindow {
|
|
|
|
return fmt.Errorf("window not found, cannot insert line[%s/%s]", line.SessionId, line.WindowId)
|
|
|
|
}
|
2022-07-12 22:50:44 +02:00
|
|
|
var maxLineId int64
|
2022-07-05 19:51:47 +02:00
|
|
|
query = `SELECT COALESCE(max(lineid), 0) FROM line WHERE sessionid = ? AND windowid = ?`
|
2022-07-02 22:31:56 +02:00
|
|
|
tx.GetWrap(&maxLineId, query, line.SessionId, line.WindowId)
|
|
|
|
line.LineId = maxLineId + 1
|
|
|
|
query = `INSERT INTO line ( sessionid, windowid, lineid, ts, userid, linetype, text, cmdid)
|
|
|
|
VALUES (:sessionid,:windowid,:lineid,:ts,:userid,:linetype,:text,:cmdid)`
|
|
|
|
tx.NamedExecWrap(query, line)
|
2022-07-08 06:39:25 +02:00
|
|
|
if cmd != nil {
|
|
|
|
cmdMap := cmd.ToMap()
|
|
|
|
query = `
|
2022-07-12 22:50:44 +02:00
|
|
|
INSERT INTO cmd ( sessionid, cmdid, remoteid, cmdstr, remotestate, termopts, status, startpk, donepk, runout, usedrows)
|
|
|
|
VALUES (:sessionid,:cmdid,:remoteid,:cmdstr,:remotestate,:termopts,:status,:startpk,:donepk,:runout,:usedrows)
|
2022-07-07 09:10:37 +02:00
|
|
|
`
|
2022-07-08 06:39:25 +02:00
|
|
|
tx.NamedExecWrap(query, cmdMap)
|
|
|
|
}
|
2022-07-07 09:10:37 +02:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-07-07 22:26:46 +02:00
|
|
|
func GetCmdById(ctx context.Context, sessionId string, cmdId string) (*CmdType, error) {
|
|
|
|
var cmd *CmdType
|
|
|
|
err := WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `SELECT * FROM cmd WHERE sessionid = ? AND cmdid = ?`
|
|
|
|
m := tx.GetMap(query, sessionId, cmdId)
|
|
|
|
cmd = CmdFromMap(m)
|
|
|
|
return nil
|
|
|
|
})
|
2022-07-07 09:10:37 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-07 22:26:46 +02:00
|
|
|
return cmd, nil
|
2022-07-07 09:10:37 +02:00
|
|
|
}
|
2022-07-08 01:29:14 +02:00
|
|
|
|
|
|
|
func UpdateCmdDonePk(ctx context.Context, donePk *packet.CmdDonePacketType) error {
|
|
|
|
if donePk == nil || donePk.CK.IsEmpty() {
|
|
|
|
return fmt.Errorf("invalid cmddone packet (no ck)")
|
|
|
|
}
|
|
|
|
return WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `UPDATE cmd SET status = ?, donepk = ? WHERE sessionid = ? AND cmdid = ?`
|
|
|
|
tx.ExecWrap(query, CmdStatusDone, quickJson(donePk), donePk.CK.GetSessionId(), donePk.CK.GetCmdId())
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func AppendCmdErrorPk(ctx context.Context, errPk *packet.CmdErrorPacketType) error {
|
|
|
|
if errPk == nil || errPk.CK.IsEmpty() {
|
|
|
|
return fmt.Errorf("invalid cmderror packet (no ck)")
|
|
|
|
}
|
|
|
|
return WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `UPDATE cmd SET runout = json_insert(runout, '$[#]', ?) WHERE sessionid = ? AND cmdid = ?`
|
|
|
|
tx.ExecWrap(query, quickJson(errPk), errPk.CK.GetSessionId(), errPk.CK.GetCmdId())
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func HangupAllRunningCmds(ctx context.Context) error {
|
|
|
|
return WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `UPDATE cmd SET status = ? WHERE status = ?`
|
|
|
|
tx.ExecWrap(query, CmdStatusHangup, CmdStatusRunning)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func HangupRunningCmdsByRemoteId(ctx context.Context, remoteId string) error {
|
|
|
|
return WithTx(ctx, func(tx *TxWrap) error {
|
|
|
|
query := `UPDATE cmd SET status = ? WHERE status = ? AND remoteid = ?`
|
|
|
|
tx.ExecWrap(query, CmdStatusHangup, CmdStatusRunning, remoteId)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|