mirror of
https://github.com/wavetermdev/waveterm.git
synced 2024-12-22 16:48:23 +01:00
241 lines
7.4 KiB
Go
241 lines
7.4 KiB
Go
// Copyright 2024, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package objectservice
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"log"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/wavetermdev/thenextwave/pkg/blockcontroller"
|
|
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
|
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
|
)
|
|
|
|
type ObjectService struct{}
|
|
|
|
const DefaultTimeout = 2 * time.Second
|
|
|
|
func parseORef(oref string) (*waveobj.ORef, error) {
|
|
fields := strings.Split(oref, ":")
|
|
if len(fields) != 2 {
|
|
return nil, fmt.Errorf("invalid object reference: %q", oref)
|
|
}
|
|
return &waveobj.ORef{OType: fields[0], OID: fields[1]}, nil
|
|
}
|
|
|
|
func (svc *ObjectService) GetObject(orefStr string) (any, error) {
|
|
oref, err := parseORef(orefStr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
obj, err := wstore.DBGetORef(ctx, *oref)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error getting object: %w", err)
|
|
}
|
|
rtn, err := waveobj.ToJsonMap(obj)
|
|
return rtn, err
|
|
}
|
|
|
|
func (svc *ObjectService) GetObjects(orefStrArr []string) (any, error) {
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
|
|
var orefArr []waveobj.ORef
|
|
for _, orefStr := range orefStrArr {
|
|
orefObj, err := parseORef(orefStr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
orefArr = append(orefArr, *orefObj)
|
|
}
|
|
return wstore.DBSelectORefs(ctx, orefArr)
|
|
}
|
|
|
|
func updatesRtn(ctx context.Context, rtnVal map[string]any) (any, error) {
|
|
updates := wstore.ContextGetUpdates(ctx)
|
|
if len(updates) == 0 {
|
|
return nil, nil
|
|
}
|
|
updateArr := make([]wstore.WaveObjUpdate, 0, len(updates))
|
|
for _, update := range updates {
|
|
updateArr = append(updateArr, update)
|
|
}
|
|
jval, err := json.Marshal(updateArr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error converting updates to JSON: %w", err)
|
|
}
|
|
if rtnVal == nil {
|
|
rtnVal = make(map[string]any)
|
|
}
|
|
rtnVal["updates"] = json.RawMessage(jval)
|
|
return rtnVal, nil
|
|
}
|
|
|
|
func (svc *ObjectService) AddTabToWorkspace(uiContext wstore.UIContext, tabName string, activateTab bool) (any, error) {
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
ctx = wstore.ContextWithUpdates(ctx)
|
|
windowData, err := wstore.DBMustGet[*wstore.Window](ctx, uiContext.WindowId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error getting window: %w", err)
|
|
}
|
|
tab, err := wstore.CreateTab(ctx, windowData.WorkspaceId, tabName)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error creating tab: %w", err)
|
|
}
|
|
if activateTab {
|
|
err = wstore.SetActiveTab(ctx, uiContext.WindowId, tab.OID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error setting active tab: %w", err)
|
|
}
|
|
}
|
|
rtn := make(map[string]any)
|
|
rtn["tabid"] = waveobj.GetOID(tab)
|
|
return updatesRtn(ctx, rtn)
|
|
}
|
|
|
|
func (svc *ObjectService) SetActiveTab(uiContext wstore.UIContext, tabId string) (any, error) {
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
ctx = wstore.ContextWithUpdates(ctx)
|
|
err := wstore.SetActiveTab(ctx, uiContext.WindowId, tabId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error setting active tab: %w", err)
|
|
}
|
|
// check all blocks in tab and start controllers (if necessary)
|
|
tab, err := wstore.DBMustGet[*wstore.Tab](ctx, tabId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error getting tab: %w", err)
|
|
}
|
|
for _, blockId := range tab.BlockIds {
|
|
blockErr := blockcontroller.StartBlockController(ctx, blockId)
|
|
if blockErr != nil {
|
|
// we don't want to fail the set active tab operation if a block controller fails to start
|
|
log.Printf("error starting block controller (blockid:%s): %w", blockId, blockErr)
|
|
continue
|
|
}
|
|
}
|
|
return updatesRtn(ctx, nil)
|
|
}
|
|
|
|
func (svc *ObjectService) CreateBlock(uiContext wstore.UIContext, blockDef *wstore.BlockDef, rtOpts *wstore.RuntimeOpts) (any, error) {
|
|
if uiContext.ActiveTabId == "" {
|
|
return nil, fmt.Errorf("no active tab")
|
|
}
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
ctx = wstore.ContextWithUpdates(ctx)
|
|
blockData, err := wstore.CreateBlock(ctx, uiContext.ActiveTabId, blockDef, rtOpts)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error creating block: %w", err)
|
|
}
|
|
if blockData.Controller != "" {
|
|
err = blockcontroller.StartBlockController(ctx, blockData.OID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error starting block controller: %w", err)
|
|
}
|
|
}
|
|
rtn := make(map[string]any)
|
|
rtn["blockid"] = blockData.OID
|
|
return updatesRtn(ctx, rtn)
|
|
}
|
|
|
|
func (svc *ObjectService) DeleteBlock(uiContext wstore.UIContext, blockId string) (any, error) {
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
ctx = wstore.ContextWithUpdates(ctx)
|
|
err := wstore.DeleteBlock(ctx, uiContext.ActiveTabId, blockId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error deleting block: %w", err)
|
|
}
|
|
blockcontroller.StopBlockController(blockId)
|
|
return updatesRtn(ctx, nil)
|
|
}
|
|
|
|
func (svc *ObjectService) CloseTab(uiContext wstore.UIContext, tabId string) (any, error) {
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
ctx = wstore.ContextWithUpdates(ctx)
|
|
window, err := wstore.DBMustGet[*wstore.Window](ctx, uiContext.WindowId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error getting window: %w", err)
|
|
}
|
|
tab, err := wstore.DBMustGet[*wstore.Tab](ctx, tabId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error getting tab: %w", err)
|
|
}
|
|
for _, blockId := range tab.BlockIds {
|
|
blockcontroller.StopBlockController(blockId)
|
|
}
|
|
err = wstore.CloseTab(ctx, window.WorkspaceId, tabId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error closing tab: %w", err)
|
|
}
|
|
if window.ActiveTabId == tabId {
|
|
ws, err := wstore.DBMustGet[*wstore.Workspace](ctx, window.WorkspaceId)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error getting workspace: %w", err)
|
|
}
|
|
var newActiveTabId string
|
|
if len(ws.TabIds) > 0 {
|
|
newActiveTabId = ws.TabIds[0]
|
|
} else {
|
|
newActiveTabId = ""
|
|
}
|
|
wstore.SetActiveTab(ctx, uiContext.WindowId, newActiveTabId)
|
|
}
|
|
return updatesRtn(ctx, nil)
|
|
}
|
|
|
|
func (svc *ObjectService) UpdateObjectMeta(uiContext wstore.UIContext, orefStr string, meta map[string]any) (any, error) {
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
ctx = wstore.ContextWithUpdates(ctx)
|
|
oref, err := parseORef(orefStr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error parsing object reference: %w", err)
|
|
}
|
|
err = wstore.UpdateObjectMeta(ctx, *oref, meta)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error updateing %q meta: %w", orefStr, err)
|
|
}
|
|
return updatesRtn(ctx, nil)
|
|
}
|
|
|
|
func (svc *ObjectService) UpdateObject(uiContext wstore.UIContext, objData map[string]any, returnUpdates bool) (any, error) {
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
|
|
defer cancelFn()
|
|
ctx = wstore.ContextWithUpdates(ctx)
|
|
|
|
oref, err := waveobj.ORefFromMap(objData)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("objData is not a valid object, requires otype and oid: %w", err)
|
|
}
|
|
found, err := wstore.DBExistsORef(ctx, *oref)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error getting object: %w", err)
|
|
}
|
|
if !found {
|
|
return nil, fmt.Errorf("object not found: %s", oref)
|
|
}
|
|
newObj, err := waveobj.FromJsonMap(objData)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error converting data to valid wave object: %w", err)
|
|
}
|
|
err = wstore.DBUpdate(ctx, newObj)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error updating object: %w", err)
|
|
}
|
|
if returnUpdates {
|
|
return updatesRtn(ctx, nil)
|
|
}
|
|
return nil, nil
|
|
}
|