2023-12-26 21:59:25 +01:00
|
|
|
// Copyright 2023, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
// in-memory storage for waveterm server
|
|
|
|
package sstore
|
|
|
|
|
|
|
|
import (
|
2024-01-12 02:34:23 +01:00
|
|
|
"fmt"
|
2023-12-26 21:59:25 +01:00
|
|
|
"log"
|
|
|
|
"sync"
|
|
|
|
|
2024-01-12 02:34:23 +01:00
|
|
|
"github.com/wavetermdev/waveterm/waveshell/pkg/packet"
|
2023-12-26 21:59:25 +01:00
|
|
|
"github.com/wavetermdev/waveterm/wavesrv/pkg/utilfn"
|
|
|
|
)
|
|
|
|
|
|
|
|
// global lock for all memory operations
|
|
|
|
// memory ops are very fast, so this is not a bottleneck
|
|
|
|
var MemLock *sync.Mutex = &sync.Mutex{}
|
|
|
|
var ScreenMemStore map[string]*ScreenMemState = make(map[string]*ScreenMemState) // map of screenid -> ScreenMemState
|
|
|
|
|
|
|
|
const (
|
|
|
|
ScreenIndicator_None = ""
|
|
|
|
ScreenIndicator_Error = "error"
|
|
|
|
ScreenIndicator_Success = "success"
|
|
|
|
ScreenIndicator_Output = "output"
|
|
|
|
)
|
|
|
|
|
|
|
|
var screenIndicatorLevels map[string]int = map[string]int{
|
|
|
|
ScreenIndicator_None: 0,
|
|
|
|
ScreenIndicator_Output: 1,
|
|
|
|
ScreenIndicator_Success: 2,
|
|
|
|
ScreenIndicator_Error: 3,
|
|
|
|
}
|
|
|
|
|
|
|
|
func dumpScreenMemStore() {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
for k, v := range ScreenMemStore {
|
|
|
|
log.Printf(" ScreenMemStore[%s] = %+v\n", k, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// returns true if i1 > i2
|
|
|
|
func isIndicatorGreater(i1 string, i2 string) bool {
|
|
|
|
return screenIndicatorLevels[i1] > screenIndicatorLevels[i2]
|
|
|
|
}
|
|
|
|
|
2024-01-12 02:34:23 +01:00
|
|
|
type OpenAICmdInfoChatStore struct {
|
|
|
|
MessageCount int `json:"messagecount"`
|
|
|
|
Messages []*packet.OpenAICmdInfoChatMessage `json:"messages"`
|
|
|
|
}
|
|
|
|
|
2023-12-26 21:59:25 +01:00
|
|
|
type ScreenMemState struct {
|
2024-01-12 02:34:23 +01:00
|
|
|
NumRunningCommands int `json:"numrunningcommands,omitempty"`
|
|
|
|
IndicatorType string `json:"indicatortype,omitempty"`
|
|
|
|
CmdInputText utilfn.StrWithPos `json:"cmdinputtext,omitempty"`
|
|
|
|
CmdInputSeqNum int `json:"cmdinputseqnum,omitempty"`
|
|
|
|
AICmdInfoChat *OpenAICmdInfoChatStore `json:"aicmdinfochat,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemDeepCopyCmdInfoChatStore(store *OpenAICmdInfoChatStore) *OpenAICmdInfoChatStore {
|
|
|
|
rtnMessages := []*packet.OpenAICmdInfoChatMessage{}
|
|
|
|
for index := 0; index < len(store.Messages); index++ {
|
|
|
|
messageToCopy := *store.Messages[index]
|
|
|
|
if messageToCopy.AssistantResponse != nil {
|
|
|
|
assistantResponseCopy := *messageToCopy.AssistantResponse
|
|
|
|
messageToCopy.AssistantResponse = &assistantResponseCopy
|
|
|
|
}
|
|
|
|
rtnMessages = append(rtnMessages, &messageToCopy)
|
|
|
|
}
|
|
|
|
rtn := &OpenAICmdInfoChatStore{MessageCount: store.MessageCount, Messages: rtnMessages}
|
|
|
|
return rtn
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemInitCmdInfoChat(screenId string) {
|
|
|
|
greetingMessagePk := &packet.OpenAICmdInfoChatMessage{
|
|
|
|
MessageID: 0,
|
|
|
|
IsAssistantResponse: true,
|
|
|
|
AssistantResponse: &packet.OpenAICmdInfoPacketOutputType{
|
|
|
|
Message: packet.OpenAICmdInfoChatGreetingMessage,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
ScreenMemStore[screenId].AICmdInfoChat = &OpenAICmdInfoChatStore{MessageCount: 1, Messages: []*packet.OpenAICmdInfoChatMessage{greetingMessagePk}}
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemClearCmdInfoChat(screenId string) {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
if ScreenMemStore[screenId] == nil {
|
|
|
|
ScreenMemStore[screenId] = &ScreenMemState{}
|
|
|
|
}
|
|
|
|
ScreenMemInitCmdInfoChat(screenId)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemAddCmdInfoChatMessage(screenId string, msg *packet.OpenAICmdInfoChatMessage) {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
if ScreenMemStore[screenId] == nil {
|
|
|
|
ScreenMemStore[screenId] = &ScreenMemState{}
|
|
|
|
}
|
|
|
|
if ScreenMemStore[screenId].AICmdInfoChat == nil {
|
|
|
|
log.Printf("AICmdInfoChat is null, creating")
|
|
|
|
ScreenMemInitCmdInfoChat(screenId)
|
|
|
|
}
|
|
|
|
|
|
|
|
CmdInfoChat := ScreenMemStore[screenId].AICmdInfoChat
|
|
|
|
CmdInfoChat.Messages = append(CmdInfoChat.Messages, msg)
|
|
|
|
CmdInfoChat.MessageCount++
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemGetCmdInfoMessageCount(screenId string) int {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
if ScreenMemStore[screenId] == nil {
|
|
|
|
ScreenMemStore[screenId] = &ScreenMemState{}
|
|
|
|
}
|
|
|
|
if ScreenMemStore[screenId].AICmdInfoChat == nil {
|
|
|
|
ScreenMemInitCmdInfoChat(screenId)
|
|
|
|
}
|
|
|
|
return ScreenMemStore[screenId].AICmdInfoChat.MessageCount
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemGetCmdInfoChat(screenId string) *OpenAICmdInfoChatStore {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
if ScreenMemStore[screenId] == nil {
|
|
|
|
ScreenMemStore[screenId] = &ScreenMemState{}
|
|
|
|
}
|
|
|
|
if ScreenMemStore[screenId].AICmdInfoChat == nil {
|
|
|
|
ScreenMemInitCmdInfoChat(screenId)
|
|
|
|
}
|
|
|
|
return ScreenMemDeepCopyCmdInfoChatStore(ScreenMemStore[screenId].AICmdInfoChat)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemUpdateCmdInfoChatMessage(screenId string, messageID int, msg *packet.OpenAICmdInfoChatMessage) error {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
if ScreenMemStore[screenId] == nil {
|
|
|
|
ScreenMemStore[screenId] = &ScreenMemState{}
|
|
|
|
}
|
|
|
|
if ScreenMemStore[screenId].AICmdInfoChat == nil {
|
|
|
|
ScreenMemInitCmdInfoChat(screenId)
|
|
|
|
}
|
|
|
|
CmdInfoChat := ScreenMemStore[screenId].AICmdInfoChat
|
|
|
|
if messageID >= 0 && messageID < len(CmdInfoChat.Messages) {
|
|
|
|
CmdInfoChat.Messages[messageID] = msg
|
|
|
|
} else {
|
|
|
|
return fmt.Errorf("ScreenMemUpdateCmdInfoChatMessage: error: Message Id out of range: %d", messageID)
|
|
|
|
}
|
|
|
|
return nil
|
2023-12-26 21:59:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemSetCmdInputText(screenId string, sp utilfn.StrWithPos, seqNum int) {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
if ScreenMemStore[screenId] == nil {
|
|
|
|
ScreenMemStore[screenId] = &ScreenMemState{}
|
|
|
|
}
|
|
|
|
if seqNum <= ScreenMemStore[screenId].CmdInputSeqNum {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ScreenMemStore[screenId].CmdInputText = sp
|
|
|
|
ScreenMemStore[screenId].CmdInputSeqNum = seqNum
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemSetNumRunningCommands(screenId string, num int) {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
if ScreenMemStore[screenId] == nil {
|
|
|
|
ScreenMemStore[screenId] = &ScreenMemState{}
|
|
|
|
}
|
|
|
|
ScreenMemStore[screenId].NumRunningCommands = num
|
|
|
|
}
|
|
|
|
|
|
|
|
func ScreenMemCombineIndicator(screenId string, indicator string) {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
if ScreenMemStore[screenId] == nil {
|
|
|
|
ScreenMemStore[screenId] = &ScreenMemState{}
|
|
|
|
}
|
|
|
|
if isIndicatorGreater(indicator, ScreenMemStore[screenId].IndicatorType) {
|
|
|
|
ScreenMemStore[screenId].IndicatorType = indicator
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// safe because we return a copy
|
|
|
|
func GetScreenMemState(screenId string) *ScreenMemState {
|
|
|
|
MemLock.Lock()
|
|
|
|
defer MemLock.Unlock()
|
|
|
|
ptr := ScreenMemStore[screenId]
|
|
|
|
if ptr == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
rtn := *ptr
|
|
|
|
return &rtn
|
|
|
|
}
|