2024-05-14 22:34:41 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
package eventbus
|
|
|
|
|
|
|
|
import (
|
2024-06-25 23:56:37 +02:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"os"
|
2024-05-14 22:34:41 +02:00
|
|
|
"sync"
|
2024-06-25 23:56:37 +02:00
|
|
|
"time"
|
2024-05-14 22:34:41 +02:00
|
|
|
|
2024-05-28 00:44:57 +02:00
|
|
|
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
2024-05-14 22:34:41 +02:00
|
|
|
)
|
|
|
|
|
2024-06-25 04:04:08 +02:00
|
|
|
const (
|
|
|
|
WSEvent_WaveObjUpdate = "waveobj:update"
|
|
|
|
WSEvent_BlockFile = "blockfile"
|
|
|
|
WSEvent_Config = "config"
|
2024-07-19 00:21:33 +02:00
|
|
|
WSEvent_UserInput = "userinput"
|
2024-06-25 04:04:08 +02:00
|
|
|
WSEvent_BlockControllerStatus = "blockcontroller:status"
|
|
|
|
WSEvent_LayoutAction = "layoutaction"
|
2024-06-25 23:56:37 +02:00
|
|
|
WSEvent_ElectronNewWindow = "electron:newwindow"
|
2024-07-23 01:39:45 +02:00
|
|
|
WSEvent_ElectronCloseWindow = "electron:closewindow"
|
2024-07-18 00:24:43 +02:00
|
|
|
WSEvent_Rpc = "rpc"
|
2024-06-25 04:04:08 +02:00
|
|
|
)
|
|
|
|
|
2024-06-12 02:42:10 +02:00
|
|
|
type WSEventType struct {
|
|
|
|
EventType string `json:"eventtype"`
|
|
|
|
ORef string `json:"oref,omitempty"`
|
|
|
|
Data any `json:"data"`
|
2024-05-14 22:34:41 +02:00
|
|
|
}
|
|
|
|
|
2024-06-14 08:54:04 +02:00
|
|
|
const (
|
2024-07-23 22:16:53 +02:00
|
|
|
FileOp_Append = "append"
|
|
|
|
FileOp_Truncate = "truncate"
|
|
|
|
FileOp_Invalidate = "invalidate"
|
2024-06-14 08:54:04 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type WSFileEventData struct {
|
|
|
|
ZoneId string `json:"zoneid"`
|
|
|
|
FileName string `json:"filename"`
|
|
|
|
FileOp string `json:"fileop"`
|
|
|
|
Data64 string `json:"data64"`
|
|
|
|
}
|
|
|
|
|
2024-05-28 00:44:57 +02:00
|
|
|
type WindowWatchData struct {
|
2024-06-12 02:42:10 +02:00
|
|
|
WindowWSCh chan any
|
|
|
|
WaveWindowId string
|
|
|
|
WatchedORefs map[waveobj.ORef]bool
|
2024-05-28 00:44:57 +02:00
|
|
|
}
|
|
|
|
|
2024-06-25 23:56:37 +02:00
|
|
|
const (
|
|
|
|
WSLayoutActionType_Insert = "insert"
|
|
|
|
WSLayoutActionType_Remove = "delete"
|
|
|
|
)
|
|
|
|
|
2024-06-25 04:04:08 +02:00
|
|
|
type WSLayoutActionData struct {
|
|
|
|
TabId string `json:"tabid"`
|
|
|
|
ActionType string `json:"actiontype"`
|
|
|
|
BlockId string `json:"blockid"`
|
2024-08-01 06:27:46 +02:00
|
|
|
NodeSize uint `json:"nodesize,omitempty"`
|
|
|
|
IndexArr []int `json:"indexarr,omitempty"`
|
2024-08-21 05:14:14 +02:00
|
|
|
Magnified bool `json:"magnified,omitempty"`
|
2024-06-25 04:04:08 +02:00
|
|
|
}
|
|
|
|
|
2024-05-14 22:34:41 +02:00
|
|
|
var globalLock = &sync.Mutex{}
|
2024-06-12 02:42:10 +02:00
|
|
|
var wsMap = make(map[string]*WindowWatchData) // websocketid => WindowWatchData
|
2024-05-14 22:34:41 +02:00
|
|
|
|
2024-06-12 02:42:10 +02:00
|
|
|
func RegisterWSChannel(connId string, windowId string, ch chan any) {
|
2024-05-14 22:34:41 +02:00
|
|
|
globalLock.Lock()
|
|
|
|
defer globalLock.Unlock()
|
2024-06-12 02:42:10 +02:00
|
|
|
wsMap[connId] = &WindowWatchData{
|
|
|
|
WindowWSCh: ch,
|
|
|
|
WaveWindowId: windowId,
|
|
|
|
WatchedORefs: make(map[waveobj.ORef]bool),
|
2024-05-28 00:44:57 +02:00
|
|
|
}
|
2024-05-14 22:34:41 +02:00
|
|
|
}
|
|
|
|
|
2024-06-12 02:42:10 +02:00
|
|
|
func UnregisterWSChannel(connId string) {
|
2024-05-14 22:34:41 +02:00
|
|
|
globalLock.Lock()
|
|
|
|
defer globalLock.Unlock()
|
2024-06-12 02:42:10 +02:00
|
|
|
delete(wsMap, connId)
|
2024-05-14 22:34:41 +02:00
|
|
|
}
|
|
|
|
|
2024-06-12 02:42:10 +02:00
|
|
|
func getWindowWatchesForWindowId(windowId string) []*WindowWatchData {
|
2024-05-28 00:44:57 +02:00
|
|
|
globalLock.Lock()
|
|
|
|
defer globalLock.Unlock()
|
2024-06-12 02:42:10 +02:00
|
|
|
var watches []*WindowWatchData
|
|
|
|
for _, wdata := range wsMap {
|
|
|
|
if wdata.WaveWindowId == windowId {
|
|
|
|
watches = append(watches, wdata)
|
2024-05-28 00:44:57 +02:00
|
|
|
}
|
|
|
|
}
|
2024-06-12 02:42:10 +02:00
|
|
|
return watches
|
2024-05-28 00:44:57 +02:00
|
|
|
}
|
|
|
|
|
2024-06-25 23:56:37 +02:00
|
|
|
// TODO fix busy wait -- but we need to wait until a new window connects back with a websocket
|
|
|
|
// returns true if the window is connected
|
|
|
|
func BusyWaitForWindowId(windowId string, timeout time.Duration) bool {
|
|
|
|
endTime := time.Now().Add(timeout)
|
|
|
|
for {
|
|
|
|
if len(getWindowWatchesForWindowId(windowId)) > 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if time.Now().After(endTime) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
time.Sleep(20 * time.Millisecond)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-12 02:42:10 +02:00
|
|
|
func getAllWatches() []*WindowWatchData {
|
|
|
|
globalLock.Lock()
|
|
|
|
defer globalLock.Unlock()
|
|
|
|
watches := make([]*WindowWatchData, 0, len(wsMap))
|
|
|
|
for _, wdata := range wsMap {
|
|
|
|
watches = append(watches, wdata)
|
2024-05-14 22:34:41 +02:00
|
|
|
}
|
2024-06-12 02:42:10 +02:00
|
|
|
return watches
|
2024-05-14 22:34:41 +02:00
|
|
|
}
|
|
|
|
|
2024-06-12 02:42:10 +02:00
|
|
|
func SendEventToWindow(windowId string, event WSEventType) {
|
|
|
|
wwdArr := getWindowWatchesForWindowId(windowId)
|
|
|
|
for _, wdata := range wwdArr {
|
|
|
|
wdata.WindowWSCh <- event
|
2024-05-14 22:34:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-12 02:42:10 +02:00
|
|
|
func SendEvent(event WSEventType) {
|
|
|
|
wwdArr := getAllWatches()
|
|
|
|
for _, wdata := range wwdArr {
|
|
|
|
wdata.WindowWSCh <- event
|
2024-05-14 22:34:41 +02:00
|
|
|
}
|
|
|
|
}
|
2024-06-25 23:56:37 +02:00
|
|
|
|
|
|
|
func SendEventToElectron(event WSEventType) {
|
|
|
|
barr, err := json.Marshal(event)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("cannot marshal electron message: %v\n", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// send to electron
|
|
|
|
log.Printf("sending event to electron: %q\n", event.EventType)
|
|
|
|
fmt.Fprintf(os.Stderr, "\nWAVESRV-EVENT:%s\n", string(barr))
|
|
|
|
}
|