2022-06-13 20:11:56 +02:00
|
|
|
package sstore
|
|
|
|
|
|
|
|
import (
|
2022-07-01 21:17:19 +02:00
|
|
|
"context"
|
2022-07-06 01:54:49 +02:00
|
|
|
"database/sql/driver"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2022-07-01 23:07:13 +02:00
|
|
|
"log"
|
2022-07-01 02:02:19 +02:00
|
|
|
"path"
|
2022-06-21 06:57:23 +02:00
|
|
|
"sync"
|
2022-06-13 20:11:56 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/google/uuid"
|
2022-07-01 19:48:14 +02:00
|
|
|
"github.com/jmoiron/sqlx"
|
2022-07-01 21:17:19 +02:00
|
|
|
"github.com/scripthaus-dev/mshell/pkg/base"
|
2022-07-01 19:48:14 +02:00
|
|
|
"github.com/scripthaus-dev/mshell/pkg/packet"
|
2022-07-01 02:02:19 +02:00
|
|
|
"github.com/scripthaus-dev/sh2-server/pkg/scbase"
|
2022-07-01 19:48:14 +02:00
|
|
|
|
|
|
|
_ "github.com/mattn/go-sqlite3"
|
2022-06-13 20:11:56 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var NextLineId = 10
|
2022-06-21 06:57:23 +02:00
|
|
|
var NextLineLock = &sync.Mutex{}
|
2022-06-13 20:11:56 +02:00
|
|
|
|
|
|
|
const LineTypeCmd = "cmd"
|
|
|
|
const LineTypeText = "text"
|
2022-07-01 19:48:14 +02:00
|
|
|
const DBFileName = "sh2.db"
|
2022-07-01 02:02:19 +02:00
|
|
|
|
2022-07-01 21:17:19 +02:00
|
|
|
const DefaultSessionName = "default"
|
|
|
|
const DefaultWindowName = "default"
|
|
|
|
const LocalRemoteName = "local"
|
|
|
|
|
2022-07-01 23:07:13 +02:00
|
|
|
const DefaultCwd = "~"
|
|
|
|
|
2022-07-01 21:17:19 +02:00
|
|
|
var globalDBLock = &sync.Mutex{}
|
|
|
|
var globalDB *sqlx.DB
|
|
|
|
var globalDBErr error
|
|
|
|
|
2022-07-01 19:48:14 +02:00
|
|
|
func GetSessionDBName() string {
|
2022-07-01 02:02:19 +02:00
|
|
|
scHome := scbase.GetScHomeDir()
|
|
|
|
return path.Join(scHome, DBFileName)
|
|
|
|
}
|
2022-06-13 20:11:56 +02:00
|
|
|
|
2022-07-01 21:17:19 +02:00
|
|
|
func GetDB() (*sqlx.DB, error) {
|
|
|
|
globalDBLock.Lock()
|
|
|
|
defer globalDBLock.Unlock()
|
|
|
|
if globalDB == nil && globalDBErr == nil {
|
|
|
|
globalDB, globalDBErr = sqlx.Open("sqlite3", GetSessionDBName())
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
2022-07-01 21:17:19 +02:00
|
|
|
return globalDB, globalDBErr
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
|
|
|
|
2022-06-21 06:57:23 +02:00
|
|
|
type SessionType struct {
|
2022-07-06 01:54:49 +02:00
|
|
|
SessionId string `json:"sessionid"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
Windows []*WindowType `json:"windows"`
|
|
|
|
Cmds []*CmdType `json:"cmds"`
|
|
|
|
Remotes []*RemoteInstance `json:"remotes"`
|
2022-07-05 07:18:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type WindowType struct {
|
|
|
|
SessionId string `json:"sessionid"`
|
|
|
|
WindowId string `json:"windowid"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
CurRemote string `json:"curremote"`
|
|
|
|
Lines []*LineType `json:"lines"`
|
|
|
|
Version int `json:"version"`
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
|
|
|
|
2022-07-06 01:54:49 +02:00
|
|
|
type RemoteState struct {
|
|
|
|
Cwd string `json:"cwd"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *RemoteState) Scan(val interface{}) error {
|
|
|
|
if strVal, ok := val.(string); ok {
|
|
|
|
if strVal == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
err := json.Unmarshal([]byte(strVal), s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return fmt.Errorf("cannot scan '%T' into RemoteState", val)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *RemoteState) Value() (driver.Value, error) {
|
|
|
|
return json.Marshal(s)
|
|
|
|
}
|
|
|
|
|
2022-07-07 09:10:37 +02:00
|
|
|
type TermOpts struct {
|
|
|
|
Rows int `json:"rows"`
|
|
|
|
Cols int `json:"cols"`
|
|
|
|
FlexRows bool `json:"flexrows,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (opts *TermOpts) Scan(val interface{}) error {
|
|
|
|
if strVal, ok := val.(string); ok {
|
|
|
|
if strVal == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
err := json.Unmarshal([]byte(strVal), opts)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return fmt.Errorf("cannot scan '%T' into TermOpts", val)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (opts *TermOpts) Value() (driver.Value, error) {
|
|
|
|
return json.Marshal(opts)
|
|
|
|
}
|
|
|
|
|
2022-07-06 01:54:49 +02:00
|
|
|
type RemoteInstance struct {
|
|
|
|
RIId string `json:"riid"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
SessionId string `json:"sessionid"`
|
|
|
|
WindowId string `json:"windowid"`
|
|
|
|
RemoteId string `json"remoteid"`
|
|
|
|
SessionScope bool `json:"sessionscope"`
|
|
|
|
State RemoteState `json:"state"`
|
2022-06-21 06:57:23 +02:00
|
|
|
}
|
|
|
|
|
2022-06-13 20:11:56 +02:00
|
|
|
type LineType struct {
|
2022-06-17 00:51:41 +02:00
|
|
|
SessionId string `json:"sessionid"`
|
|
|
|
WindowId string `json:"windowid"`
|
2022-06-13 20:11:56 +02:00
|
|
|
LineId int `json:"lineid"`
|
|
|
|
Ts int64 `json:"ts"`
|
|
|
|
UserId string `json:"userid"`
|
|
|
|
LineType string `json:"linetype"`
|
|
|
|
Text string `json:"text,omitempty"`
|
|
|
|
CmdId string `json:"cmdid,omitempty"`
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type RemoteType struct {
|
2022-07-02 02:38:36 +02:00
|
|
|
RemoteId string `json:"remoteid"`
|
|
|
|
RemoteType string `json:"remotetype"`
|
|
|
|
RemoteName string `json:"remotename"`
|
|
|
|
AutoConnect bool `json:"autoconnect"`
|
|
|
|
|
|
|
|
// type=ssh options
|
|
|
|
SSHHost string `json:"sshhost"`
|
|
|
|
SSHOpts string `json:"sshopts"`
|
|
|
|
SSHIdentity string `json:"sshidentity"`
|
|
|
|
SSHUser string `json:"sshuser"`
|
|
|
|
|
|
|
|
// runtime data
|
|
|
|
LastConnectTs int64 `json:"lastconnectts"`
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type CmdType struct {
|
2022-07-07 09:10:37 +02:00
|
|
|
SessionId string `json:"sessionid"`
|
|
|
|
CmdId string `json:"cmdid"`
|
|
|
|
RemoteId string `json:"remoteid"`
|
|
|
|
CmdStr string `json:"cmdstr"`
|
|
|
|
RemoteState RemoteState `json:"remotestate"`
|
|
|
|
TermOpts TermOpts `json:"termopts"`
|
|
|
|
Status string `json:"status"`
|
|
|
|
StartPk *packet.CmdStartPacketType `json:"startpk"`
|
|
|
|
DonePk *packet.CmdDonePacketType `json:"donepk"`
|
|
|
|
RunOut []packet.PacketType `json:"runout"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func quickJson(v interface{}) string {
|
|
|
|
if v == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
barr, _ := json.Marshal(v)
|
|
|
|
return string(barr)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cmd *CmdType) ToMap() map[string]interface{} {
|
|
|
|
rtn := make(map[string]interface{})
|
|
|
|
rtn["sessionid"] = cmd.SessionId
|
|
|
|
rtn["cmdid"] = cmd.CmdId
|
|
|
|
rtn["remoteid"] = cmd.RemoteId
|
|
|
|
rtn["cmdstr"] = cmd.CmdStr
|
|
|
|
rtn["remotestate"] = quickJson(cmd.RemoteState)
|
|
|
|
rtn["termopts"] = quickJson(cmd.TermOpts)
|
|
|
|
rtn["status"] = cmd.Status
|
|
|
|
rtn["startpk"] = quickJson(cmd.StartPk)
|
|
|
|
rtn["donepk"] = quickJson(cmd.DonePk)
|
|
|
|
rtn["runout"] = quickJson(cmd.RunOut)
|
|
|
|
return rtn
|
|
|
|
}
|
|
|
|
|
|
|
|
func quickSetStr(strVal *string, m map[string]interface{}, name string) {
|
|
|
|
v, ok := m[name]
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
str, ok := v.(string)
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
*strVal = str
|
|
|
|
}
|
|
|
|
|
|
|
|
func quickSetJson(ptr interface{}, m map[string]interface{}, name string) {
|
|
|
|
v, ok := m[name]
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
str, ok := v.(string)
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if str == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
json.Unmarshal([]byte(str), ptr)
|
|
|
|
}
|
|
|
|
|
|
|
|
func CmdFromMap(m map[string]interface{}) *CmdType {
|
|
|
|
var cmd CmdType
|
|
|
|
quickSetStr(&cmd.SessionId, m, "sessionid")
|
|
|
|
quickSetStr(&cmd.CmdId, m, "cmdid")
|
|
|
|
quickSetStr(&cmd.RemoteId, m, "remoteid")
|
|
|
|
quickSetStr(&cmd.CmdStr, m, "cmdstr")
|
|
|
|
quickSetJson(&cmd.RemoteState, m, "remotestate")
|
|
|
|
quickSetJson(&cmd.TermOpts, m, "termopts")
|
|
|
|
quickSetStr(&cmd.Status, m, "status")
|
|
|
|
quickSetJson(&cmd.StartPk, m, "startpk")
|
|
|
|
quickSetJson(&cmd.DonePk, m, "donepk")
|
|
|
|
quickSetJson(&cmd.RunOut, m, "runout")
|
|
|
|
return &cmd
|
2022-06-13 20:11:56 +02:00
|
|
|
}
|
|
|
|
|
2022-07-05 19:51:47 +02:00
|
|
|
func makeNewLineCmd(sessionId string, windowId string, userId string) *LineType {
|
2022-06-13 20:11:56 +02:00
|
|
|
rtn := &LineType{}
|
2022-06-17 00:51:41 +02:00
|
|
|
rtn.SessionId = sessionId
|
|
|
|
rtn.WindowId = windowId
|
2022-06-13 20:11:56 +02:00
|
|
|
rtn.Ts = time.Now().UnixMilli()
|
2022-07-05 19:51:47 +02:00
|
|
|
rtn.UserId = userId
|
2022-06-13 20:11:56 +02:00
|
|
|
rtn.LineType = LineTypeCmd
|
|
|
|
rtn.CmdId = uuid.New().String()
|
|
|
|
return rtn
|
|
|
|
}
|
|
|
|
|
2022-07-05 19:51:47 +02:00
|
|
|
func makeNewLineText(sessionId string, windowId string, userId string, text string) *LineType {
|
2022-06-13 20:11:56 +02:00
|
|
|
rtn := &LineType{}
|
2022-06-17 00:51:41 +02:00
|
|
|
rtn.SessionId = sessionId
|
|
|
|
rtn.WindowId = windowId
|
2022-06-13 20:11:56 +02:00
|
|
|
rtn.Ts = time.Now().UnixMilli()
|
2022-07-05 19:51:47 +02:00
|
|
|
rtn.UserId = userId
|
2022-06-13 20:11:56 +02:00
|
|
|
rtn.LineType = LineTypeText
|
|
|
|
rtn.Text = text
|
|
|
|
return rtn
|
|
|
|
}
|
2022-06-21 06:57:23 +02:00
|
|
|
|
2022-07-05 19:51:47 +02:00
|
|
|
func AddCommentLine(ctx context.Context, sessionId string, windowId string, userId string, commentText string) (*LineType, error) {
|
|
|
|
rtnLine := makeNewLineText(sessionId, windowId, userId, commentText)
|
|
|
|
err := InsertLine(ctx, rtnLine)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return rtnLine, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func AddCmdLine(ctx context.Context, sessionId string, windowId string, userId string) (*LineType, error) {
|
|
|
|
rtnLine := makeNewLineCmd(sessionId, windowId, userId)
|
|
|
|
err := InsertLine(ctx, rtnLine)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return rtnLine, nil
|
|
|
|
}
|
|
|
|
|
2022-06-21 06:57:23 +02:00
|
|
|
func GetNextLine() int {
|
|
|
|
NextLineLock.Lock()
|
|
|
|
defer NextLineLock.Unlock()
|
|
|
|
rtn := NextLineId
|
|
|
|
NextLineId++
|
|
|
|
return rtn
|
|
|
|
}
|
2022-07-01 21:17:19 +02:00
|
|
|
|
|
|
|
func EnsureLocalRemote(ctx context.Context) error {
|
|
|
|
remoteId, err := base.GetRemoteId()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
remote, err := GetRemoteById(ctx, remoteId)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if remote != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// create the local remote
|
|
|
|
localRemote := &RemoteType{
|
2022-07-02 02:38:36 +02:00
|
|
|
RemoteId: remoteId,
|
|
|
|
RemoteType: "ssh",
|
|
|
|
RemoteName: LocalRemoteName,
|
|
|
|
AutoConnect: true,
|
2022-07-01 21:17:19 +02:00
|
|
|
}
|
|
|
|
err = InsertRemote(ctx, localRemote)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-07-01 23:07:13 +02:00
|
|
|
log.Printf("[db] added remote '%s', id=%s\n", localRemote.RemoteName, localRemote.RemoteId)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-07-01 23:45:33 +02:00
|
|
|
func EnsureDefaultSession(ctx context.Context) (*SessionType, error) {
|
2022-07-01 23:07:13 +02:00
|
|
|
session, err := GetSessionByName(ctx, DefaultSessionName)
|
|
|
|
if err != nil {
|
2022-07-01 23:45:33 +02:00
|
|
|
return nil, err
|
2022-07-01 23:07:13 +02:00
|
|
|
}
|
|
|
|
if session != nil {
|
2022-07-01 23:45:33 +02:00
|
|
|
return session, nil
|
2022-07-01 23:07:13 +02:00
|
|
|
}
|
|
|
|
err = InsertSessionWithName(ctx, DefaultSessionName)
|
|
|
|
if err != nil {
|
2022-07-01 23:45:33 +02:00
|
|
|
return nil, err
|
2022-07-01 23:07:13 +02:00
|
|
|
}
|
2022-07-01 23:45:33 +02:00
|
|
|
return GetSessionByName(ctx, DefaultSessionName)
|
2022-07-01 21:17:19 +02:00
|
|
|
}
|