mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-04 18:59:08 +01:00
checkpoint on domain sockets + update background colors + transparency (#160)
This commit is contained in:
parent
f1837d6fae
commit
9df9c99fbd
@ -8,11 +8,12 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"reflect"
|
"reflect"
|
||||||
"sort"
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/wavetermdev/thenextwave/pkg/service"
|
"github.com/wavetermdev/thenextwave/pkg/service"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/tsgen"
|
"github.com/wavetermdev/thenextwave/pkg/tsgen"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/util/utilfn"
|
"github.com/wavetermdev/thenextwave/pkg/util/utilfn"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshrpc/wshserver"
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
)
|
)
|
||||||
|
|
||||||
func generateTypesFile(tsTypesMap map[reflect.Type]string) error {
|
func generateTypesFile(tsTypesMap map[reflect.Type]string) error {
|
||||||
@ -43,6 +44,10 @@ func generateTypesFile(tsTypesMap map[reflect.Type]string) error {
|
|||||||
return iname < jname
|
return iname < jname
|
||||||
})
|
})
|
||||||
for _, key := range keys {
|
for _, key := range keys {
|
||||||
|
// don't output generic types
|
||||||
|
if strings.Index(key.Name(), "[") != -1 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
tsCode := tsTypesMap[key]
|
tsCode := tsTypesMap[key]
|
||||||
istr := utilfn.IndentString(" ", tsCode)
|
istr := utilfn.IndentString(" ", tsCode)
|
||||||
fmt.Fprint(fd, istr)
|
fmt.Fprint(fd, istr)
|
||||||
@ -79,16 +84,17 @@ func generateWshServerFile(tsTypeMap map[reflect.Type]string) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
defer fd.Close()
|
defer fd.Close()
|
||||||
|
declMap := wshrpc.GenerateWshCommandDeclMap()
|
||||||
fmt.Fprintf(os.Stderr, "generating wshserver file to %s\n", fd.Name())
|
fmt.Fprintf(os.Stderr, "generating wshserver file to %s\n", fd.Name())
|
||||||
fmt.Fprintf(fd, "// Copyright 2024, Command Line Inc.\n")
|
fmt.Fprintf(fd, "// Copyright 2024, Command Line Inc.\n")
|
||||||
fmt.Fprintf(fd, "// SPDX-License-Identifier: Apache-2.0\n\n")
|
fmt.Fprintf(fd, "// SPDX-License-Identifier: Apache-2.0\n\n")
|
||||||
fmt.Fprintf(fd, "// generated by cmd/generate/main-generate.go\n\n")
|
fmt.Fprintf(fd, "// generated by cmd/generate/main-generate.go\n\n")
|
||||||
fmt.Fprintf(fd, "import * as WOS from \"./wos\";\n\n")
|
fmt.Fprintf(fd, "import * as WOS from \"./wos\";\n\n")
|
||||||
orderedKeys := utilfn.GetOrderedMapKeys(wshserver.WshServerCommandToDeclMap)
|
orderedKeys := utilfn.GetOrderedMapKeys(declMap)
|
||||||
fmt.Fprintf(fd, "// WshServerCommandToDeclMap\n")
|
fmt.Fprintf(fd, "// WshServerCommandToDeclMap\n")
|
||||||
fmt.Fprintf(fd, "class WshServerType {\n")
|
fmt.Fprintf(fd, "class WshServerType {\n")
|
||||||
for _, methodDecl := range orderedKeys {
|
for _, methodDecl := range orderedKeys {
|
||||||
methodDecl := wshserver.WshServerCommandToDeclMap[methodDecl]
|
methodDecl := declMap[methodDecl]
|
||||||
methodStr := tsgen.GenerateWshServerMethod(methodDecl, tsTypeMap)
|
methodStr := tsgen.GenerateWshServerMethod(methodDecl, tsTypeMap)
|
||||||
fmt.Fprint(fd, methodStr)
|
fmt.Fprint(fd, methodStr)
|
||||||
fmt.Fprintf(fd, "\n")
|
fmt.Fprintf(fd, "\n")
|
||||||
|
@ -8,11 +8,10 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
|
|
||||||
"github.com/wavetermdev/thenextwave/pkg/util/utilfn"
|
"github.com/wavetermdev/thenextwave/pkg/util/utilfn"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshrpc/wshserver"
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func genMethod_ResponseStream(fd *os.File, methodDecl *wshserver.WshServerMethodDecl) {
|
func genMethod_ResponseStream(fd *os.File, methodDecl *wshrpc.WshRpcMethodDecl) {
|
||||||
fmt.Fprintf(fd, "// command %q, wshserver.%s\n", methodDecl.Command, methodDecl.MethodName)
|
fmt.Fprintf(fd, "// command %q, wshserver.%s\n", methodDecl.Command, methodDecl.MethodName)
|
||||||
var dataType string
|
var dataType string
|
||||||
dataVarName := "nil"
|
dataVarName := "nil"
|
||||||
@ -29,7 +28,7 @@ func genMethod_ResponseStream(fd *os.File, methodDecl *wshserver.WshServerMethod
|
|||||||
fmt.Fprintf(fd, "}\n\n")
|
fmt.Fprintf(fd, "}\n\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
func genMethod_Call(fd *os.File, methodDecl *wshserver.WshServerMethodDecl) {
|
func genMethod_Call(fd *os.File, methodDecl *wshrpc.WshRpcMethodDecl) {
|
||||||
fmt.Fprintf(fd, "// command %q, wshserver.%s\n", methodDecl.Command, methodDecl.MethodName)
|
fmt.Fprintf(fd, "// command %q, wshserver.%s\n", methodDecl.Command, methodDecl.MethodName)
|
||||||
var dataType string
|
var dataType string
|
||||||
dataVarName := "nil"
|
dataVarName := "nil"
|
||||||
@ -70,14 +69,14 @@ func main() {
|
|||||||
fmt.Fprintf(fd, " \"github.com/wavetermdev/thenextwave/pkg/wshutil\"\n")
|
fmt.Fprintf(fd, " \"github.com/wavetermdev/thenextwave/pkg/wshutil\"\n")
|
||||||
fmt.Fprintf(fd, " \"github.com/wavetermdev/thenextwave/pkg/wshrpc\"\n")
|
fmt.Fprintf(fd, " \"github.com/wavetermdev/thenextwave/pkg/wshrpc\"\n")
|
||||||
fmt.Fprintf(fd, " \"github.com/wavetermdev/thenextwave/pkg/waveobj\"\n")
|
fmt.Fprintf(fd, " \"github.com/wavetermdev/thenextwave/pkg/waveobj\"\n")
|
||||||
fmt.Fprintf(fd, " \"github.com/wavetermdev/thenextwave/pkg/waveai\"\n")
|
|
||||||
fmt.Fprintf(fd, ")\n\n")
|
fmt.Fprintf(fd, ")\n\n")
|
||||||
|
|
||||||
for _, key := range utilfn.GetOrderedMapKeys(wshserver.WshServerCommandToDeclMap) {
|
wshDeclMap := wshrpc.GenerateWshCommandDeclMap()
|
||||||
methodDecl := wshserver.WshServerCommandToDeclMap[key]
|
for _, key := range utilfn.GetOrderedMapKeys(wshDeclMap) {
|
||||||
if methodDecl.CommandType == wshutil.RpcType_ResponseStream {
|
methodDecl := wshDeclMap[key]
|
||||||
|
if methodDecl.CommandType == wshrpc.RpcType_ResponseStream {
|
||||||
genMethod_ResponseStream(fd, methodDecl)
|
genMethod_ResponseStream(fd, methodDecl)
|
||||||
} else if methodDecl.CommandType == wshutil.RpcType_Call {
|
} else if methodDecl.CommandType == wshrpc.RpcType_Call {
|
||||||
genMethod_Call(fd, methodDecl)
|
genMethod_Call(fd, methodDecl)
|
||||||
} else {
|
} else {
|
||||||
panic("unsupported command type " + methodDecl.CommandType)
|
panic("unsupported command type " + methodDecl.CommandType)
|
||||||
|
@ -43,7 +43,7 @@ func runReadFile(cmd *cobra.Command, args []string) {
|
|||||||
fmt.Fprintf(os.Stderr, "error resolving oref: %v\r\n", err)
|
fmt.Fprintf(os.Stderr, "error resolving oref: %v\r\n", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
resp64, err := wshclient.ReadFile(RpcClient, wshrpc.CommandFileData{ZoneId: fullORef.OID, FileName: args[1]}, &wshrpc.WshRpcCommandOpts{Timeout: 5000})
|
resp64, err := wshclient.FileReadCommand(RpcClient, wshrpc.CommandFileData{ZoneId: fullORef.OID, FileName: args[1]}, &wshrpc.WshRpcCommandOpts{Timeout: 5000})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Fprintf(os.Stderr, "error reading file: %v\r\n", err)
|
fmt.Fprintf(os.Stderr, "error reading file: %v\r\n", err)
|
||||||
return
|
return
|
||||||
|
@ -34,6 +34,7 @@ const waveSrvReady: Promise<boolean> = new Promise((resolve, _) => {
|
|||||||
});
|
});
|
||||||
let globalIsQuitting = false;
|
let globalIsQuitting = false;
|
||||||
let globalIsStarting = true;
|
let globalIsStarting = true;
|
||||||
|
let globalIsRelaunching = false;
|
||||||
|
|
||||||
const isDev = !electronApp.isPackaged;
|
const isDev = !electronApp.isPackaged;
|
||||||
const isDevVite = isDev && process.env.ELECTRON_RENDERER_URL;
|
const isDevVite = isDev && process.env.ELECTRON_RENDERER_URL;
|
||||||
@ -214,7 +215,8 @@ async function handleWSEvent(evtMsg: WSEventType) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const clientData = await services.ClientService.GetClientData();
|
const clientData = await services.ClientService.GetClientData();
|
||||||
const newWin = createBrowserWindow(clientData.oid, windowData);
|
const settings = await services.FileService.GetSettingsConfig();
|
||||||
|
const newWin = createBrowserWindow(clientData.oid, windowData, settings);
|
||||||
await newWin.readyPromise;
|
await newWin.readyPromise;
|
||||||
newWin.show();
|
newWin.show();
|
||||||
} else if (evtMsg.eventtype == "electron:closewindow") {
|
} else if (evtMsg.eventtype == "electron:closewindow") {
|
||||||
@ -290,7 +292,11 @@ function shFrameNavHandler(event: Electron.Event<Electron.WebContentsWillFrameNa
|
|||||||
|
|
||||||
// note, this does not *show* the window.
|
// note, this does not *show* the window.
|
||||||
// to show, await win.readyPromise and then win.show()
|
// to show, await win.readyPromise and then win.show()
|
||||||
function createBrowserWindow(clientId: string, waveWindow: WaveWindow): WaveBrowserWindow {
|
function createBrowserWindow(
|
||||||
|
clientId: string,
|
||||||
|
waveWindow: WaveWindow,
|
||||||
|
settings: SettingsConfigType
|
||||||
|
): WaveBrowserWindow {
|
||||||
let winBounds = {
|
let winBounds = {
|
||||||
x: waveWindow.pos.x,
|
x: waveWindow.pos.x,
|
||||||
y: waveWindow.pos.y,
|
y: waveWindow.pos.y,
|
||||||
@ -298,7 +304,7 @@ function createBrowserWindow(clientId: string, waveWindow: WaveWindow): WaveBrow
|
|||||||
height: waveWindow.winsize.height,
|
height: waveWindow.winsize.height,
|
||||||
};
|
};
|
||||||
winBounds = ensureBoundsAreVisible(winBounds);
|
winBounds = ensureBoundsAreVisible(winBounds);
|
||||||
const bwin = new electron.BrowserWindow({
|
const winOpts: Electron.BrowserWindowConstructorOptions = {
|
||||||
titleBarStyle: "hiddenInset",
|
titleBarStyle: "hiddenInset",
|
||||||
x: winBounds.x,
|
x: winBounds.x,
|
||||||
y: winBounds.y,
|
y: winBounds.y,
|
||||||
@ -316,8 +322,14 @@ function createBrowserWindow(clientId: string, waveWindow: WaveWindow): WaveBrow
|
|||||||
},
|
},
|
||||||
show: false,
|
show: false,
|
||||||
autoHideMenuBar: true,
|
autoHideMenuBar: true,
|
||||||
backgroundColor: "#000000",
|
};
|
||||||
});
|
const isTransparent = settings?.window?.transparent ?? true;
|
||||||
|
if (isTransparent) {
|
||||||
|
winOpts.transparent = true;
|
||||||
|
} else {
|
||||||
|
winOpts.backgroundColor = "#222222";
|
||||||
|
}
|
||||||
|
const bwin = new electron.BrowserWindow(winOpts);
|
||||||
(bwin as any).waveWindowId = waveWindow.oid;
|
(bwin as any).waveWindowId = waveWindow.oid;
|
||||||
let readyResolve: (value: void) => void;
|
let readyResolve: (value: void) => void;
|
||||||
(bwin as any).readyPromise = new Promise((resolve, _) => {
|
(bwin as any).readyPromise = new Promise((resolve, _) => {
|
||||||
@ -519,7 +531,8 @@ electron.ipcMain.on("getEnv", (event, varName) => {
|
|||||||
async function createNewWaveWindow() {
|
async function createNewWaveWindow() {
|
||||||
const clientData = await services.ClientService.GetClientData();
|
const clientData = await services.ClientService.GetClientData();
|
||||||
const newWindow = await services.ClientService.MakeWindow();
|
const newWindow = await services.ClientService.MakeWindow();
|
||||||
const newBrowserWindow = createBrowserWindow(clientData.oid, newWindow);
|
const settings = await services.FileService.GetSettingsConfig();
|
||||||
|
const newBrowserWindow = createBrowserWindow(clientData.oid, newWindow, settings);
|
||||||
newBrowserWindow.show();
|
newBrowserWindow.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -616,6 +629,12 @@ function makeAppMenu() {
|
|||||||
{
|
{
|
||||||
role: "forceReload",
|
role: "forceReload",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: "Relaunch All Windows",
|
||||||
|
click: () => {
|
||||||
|
relaunchBrowserWindows();
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
role: "toggleDevTools",
|
role: "toggleDevTools",
|
||||||
},
|
},
|
||||||
@ -663,6 +682,9 @@ function makeAppMenu() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
electronApp.on("window-all-closed", () => {
|
electronApp.on("window-all-closed", () => {
|
||||||
|
if (globalIsRelaunching) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (unamePlatform !== "darwin") {
|
if (unamePlatform !== "darwin") {
|
||||||
electronApp.quit();
|
electronApp.quit();
|
||||||
}
|
}
|
||||||
@ -857,6 +879,36 @@ async function configureAutoUpdater() {
|
|||||||
}
|
}
|
||||||
// ====== AUTO-UPDATER ====== //
|
// ====== AUTO-UPDATER ====== //
|
||||||
|
|
||||||
|
async function relaunchBrowserWindows() {
|
||||||
|
globalIsRelaunching = true;
|
||||||
|
const windows = electron.BrowserWindow.getAllWindows();
|
||||||
|
for (const window of windows) {
|
||||||
|
window.removeAllListeners();
|
||||||
|
window.close();
|
||||||
|
}
|
||||||
|
globalIsRelaunching = false;
|
||||||
|
|
||||||
|
const clientData = await services.ClientService.GetClientData();
|
||||||
|
const settings = await services.FileService.GetSettingsConfig();
|
||||||
|
const wins: WaveBrowserWindow[] = [];
|
||||||
|
for (const windowId of clientData.windowids.slice().reverse()) {
|
||||||
|
const windowData: WaveWindow = (await services.ObjectService.GetObject("window:" + windowId)) as WaveWindow;
|
||||||
|
if (windowData == null) {
|
||||||
|
services.WindowService.CloseWindow(windowId).catch((e) => {
|
||||||
|
/* ignore */
|
||||||
|
});
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const win = createBrowserWindow(clientData.oid, windowData, settings);
|
||||||
|
wins.push(win);
|
||||||
|
}
|
||||||
|
for (const win of wins) {
|
||||||
|
await win.readyPromise;
|
||||||
|
console.log("show", win.waveWindowId);
|
||||||
|
win.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async function appMain() {
|
async function appMain() {
|
||||||
const startTs = Date.now();
|
const startTs = Date.now();
|
||||||
const instanceLock = electronApp.requestSingleInstanceLock();
|
const instanceLock = electronApp.requestSingleInstanceLock();
|
||||||
@ -877,27 +929,8 @@ async function appMain() {
|
|||||||
}
|
}
|
||||||
const ready = await waveSrvReady;
|
const ready = await waveSrvReady;
|
||||||
console.log("wavesrv ready signal received", ready, Date.now() - startTs, "ms");
|
console.log("wavesrv ready signal received", ready, Date.now() - startTs, "ms");
|
||||||
console.log("get client data");
|
|
||||||
const clientData = await services.ClientService.GetClientData();
|
|
||||||
console.log("client data ready");
|
|
||||||
await electronApp.whenReady();
|
await electronApp.whenReady();
|
||||||
const wins: WaveBrowserWindow[] = [];
|
relaunchBrowserWindows();
|
||||||
for (const windowId of clientData.windowids.slice().reverse()) {
|
|
||||||
const windowData: WaveWindow = (await services.ObjectService.GetObject("window:" + windowId)) as WaveWindow;
|
|
||||||
if (windowData == null) {
|
|
||||||
services.WindowService.CloseWindow(windowId).catch((e) => {
|
|
||||||
/* ignore */
|
|
||||||
});
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
const win = createBrowserWindow(clientData.oid, windowData);
|
|
||||||
wins.push(win);
|
|
||||||
}
|
|
||||||
for (const win of wins) {
|
|
||||||
await win.readyPromise;
|
|
||||||
console.log("show", win.waveWindowId);
|
|
||||||
win.show();
|
|
||||||
}
|
|
||||||
configureAutoUpdater();
|
configureAutoUpdater();
|
||||||
globalIsStarting = false;
|
globalIsStarting = false;
|
||||||
|
|
||||||
|
@ -14,6 +14,8 @@ body {
|
|||||||
font: var(--base-font);
|
font: var(--base-font);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
-webkit-font-smoothing: auto;
|
-webkit-font-smoothing: auto;
|
||||||
|
backface-visibility: hidden;
|
||||||
|
transform: translateZ(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
*::-webkit-scrollbar {
|
*::-webkit-scrollbar {
|
||||||
|
@ -16,6 +16,7 @@ import { HTML5Backend } from "react-dnd-html5-backend";
|
|||||||
import { CenteredDiv } from "./element/quickelems";
|
import { CenteredDiv } from "./element/quickelems";
|
||||||
|
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
|
import Color from "color";
|
||||||
import "overlayscrollbars/overlayscrollbars.css";
|
import "overlayscrollbars/overlayscrollbars.css";
|
||||||
import "./app.less";
|
import "./app.less";
|
||||||
|
|
||||||
@ -200,6 +201,31 @@ function switchBlock(tabId: string, offsetX: number, offsetY: number) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function AppSettingsUpdater() {
|
||||||
|
const settings = jotai.useAtomValue(atoms.settingsConfigAtom);
|
||||||
|
React.useEffect(() => {
|
||||||
|
let isTransparent = settings?.window?.transparent ?? true;
|
||||||
|
let opacity = util.boundNumber(settings?.window?.opacity ?? 0.8, 0, 1);
|
||||||
|
let baseBgColor = settings?.window?.bgcolor;
|
||||||
|
console.log("window settings", settings.window);
|
||||||
|
|
||||||
|
if (isTransparent) {
|
||||||
|
document.body.classList.add("is-transparent");
|
||||||
|
const rootStyles = getComputedStyle(document.documentElement);
|
||||||
|
if (baseBgColor == null) {
|
||||||
|
baseBgColor = rootStyles.getPropertyValue("--main-bg-color").trim();
|
||||||
|
}
|
||||||
|
const color = new Color(baseBgColor);
|
||||||
|
const rgbaColor = color.alpha(opacity).string();
|
||||||
|
document.body.style.backgroundColor = rgbaColor;
|
||||||
|
} else {
|
||||||
|
document.body.classList.remove("is-transparent");
|
||||||
|
document.body.style.opacity = null;
|
||||||
|
}
|
||||||
|
}, [settings?.window]);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
const AppInner = () => {
|
const AppInner = () => {
|
||||||
const client = jotai.useAtomValue(atoms.client);
|
const client = jotai.useAtomValue(atoms.client);
|
||||||
const windowData = jotai.useAtomValue(atoms.waveWindow);
|
const windowData = jotai.useAtomValue(atoms.waveWindow);
|
||||||
@ -251,6 +277,7 @@ const AppInner = () => {
|
|||||||
const isFullScreen = jotai.useAtomValue(atoms.isFullScreen);
|
const isFullScreen = jotai.useAtomValue(atoms.isFullScreen);
|
||||||
return (
|
return (
|
||||||
<div className={clsx("mainapp", PLATFORM, { fullscreen: isFullScreen })} onContextMenu={handleContextMenu}>
|
<div className={clsx("mainapp", PLATFORM, { fullscreen: isFullScreen })} onContextMenu={handleContextMenu}>
|
||||||
|
<AppSettingsUpdater />
|
||||||
<DndProvider backend={HTML5Backend}>
|
<DndProvider backend={HTML5Backend}>
|
||||||
<Workspace />
|
<Workspace />
|
||||||
</DndProvider>
|
</DndProvider>
|
||||||
|
@ -59,7 +59,7 @@
|
|||||||
padding: 2px;
|
padding: 2px;
|
||||||
|
|
||||||
.block-frame-default-inner {
|
.block-frame-default-inner {
|
||||||
background-color: rgba(0, 0, 0, 0.5);
|
background-color: var(--block-bg-color);
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
|
@ -7,14 +7,19 @@ import * as WOS from "./wos";
|
|||||||
|
|
||||||
// WshServerCommandToDeclMap
|
// WshServerCommandToDeclMap
|
||||||
class WshServerType {
|
class WshServerType {
|
||||||
// command "controller:input" [call]
|
// command "authenticate" [call]
|
||||||
BlockInputCommand(data: CommandBlockInputData, opts?: WshRpcCommandOpts): Promise<void> {
|
AuthenticateCommand(data: string, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
return WOS.wshServerRpcHelper_call("controller:input", data, opts);
|
return WOS.wshServerRpcHelper_call("authenticate", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "controller:restart" [call]
|
// command "controllerinput" [call]
|
||||||
BlockRestartCommand(data: CommandBlockRestartData, opts?: WshRpcCommandOpts): Promise<void> {
|
ControllerInputCommand(data: CommandBlockInputData, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
return WOS.wshServerRpcHelper_call("controller:restart", data, opts);
|
return WOS.wshServerRpcHelper_call("controllerinput", data, opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "controllerrestart" [call]
|
||||||
|
ControllerRestartCommand(data: CommandBlockRestartData, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
|
return WOS.wshServerRpcHelper_call("controllerrestart", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "createblock" [call]
|
// command "createblock" [call]
|
||||||
@ -27,24 +32,49 @@ class WshServerType {
|
|||||||
return WOS.wshServerRpcHelper_call("deleteblock", data, opts);
|
return WOS.wshServerRpcHelper_call("deleteblock", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "file:append" [call]
|
// command "eventpublish" [call]
|
||||||
AppendFileCommand(data: CommandFileData, opts?: WshRpcCommandOpts): Promise<void> {
|
EventPublishCommand(data: WaveEvent, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
return WOS.wshServerRpcHelper_call("file:append", data, opts);
|
return WOS.wshServerRpcHelper_call("eventpublish", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "file:appendijson" [call]
|
// command "eventrecv" [call]
|
||||||
AppendIJsonCommand(data: CommandAppendIJsonData, opts?: WshRpcCommandOpts): Promise<void> {
|
EventRecvCommand(data: WaveEvent, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
return WOS.wshServerRpcHelper_call("file:appendijson", data, opts);
|
return WOS.wshServerRpcHelper_call("eventrecv", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "file:read" [call]
|
// command "eventsub" [call]
|
||||||
ReadFile(data: CommandFileData, opts?: WshRpcCommandOpts): Promise<string> {
|
EventSubCommand(data: SubscriptionRequest, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
return WOS.wshServerRpcHelper_call("file:read", data, opts);
|
return WOS.wshServerRpcHelper_call("eventsub", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "file:write" [call]
|
// command "eventunsub" [call]
|
||||||
WriteFile(data: CommandFileData, opts?: WshRpcCommandOpts): Promise<void> {
|
EventUnsubCommand(data: SubscriptionRequest, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
return WOS.wshServerRpcHelper_call("file:write", data, opts);
|
return WOS.wshServerRpcHelper_call("eventunsub", data, opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "eventunsuball" [call]
|
||||||
|
EventUnsubAllCommand(opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
|
return WOS.wshServerRpcHelper_call("eventunsuball", null, opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "fileappend" [call]
|
||||||
|
FileAppendCommand(data: CommandFileData, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
|
return WOS.wshServerRpcHelper_call("fileappend", data, opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "fileappendijson" [call]
|
||||||
|
FileAppendIJsonCommand(data: CommandAppendIJsonData, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
|
return WOS.wshServerRpcHelper_call("fileappendijson", data, opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "fileread" [call]
|
||||||
|
FileReadCommand(data: CommandFileData, opts?: WshRpcCommandOpts): Promise<string> {
|
||||||
|
return WOS.wshServerRpcHelper_call("fileread", data, opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "filewrite" [call]
|
||||||
|
FileWriteCommand(data: CommandFileData, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
|
return WOS.wshServerRpcHelper_call("filewrite", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "getmeta" [call]
|
// command "getmeta" [call]
|
||||||
@ -68,18 +98,18 @@ class WshServerType {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// command "setview" [call]
|
// command "setview" [call]
|
||||||
BlockSetViewCommand(data: CommandBlockSetViewData, opts?: WshRpcCommandOpts): Promise<void> {
|
SetViewCommand(data: CommandBlockSetViewData, opts?: WshRpcCommandOpts): Promise<void> {
|
||||||
return WOS.wshServerRpcHelper_call("setview", data, opts);
|
return WOS.wshServerRpcHelper_call("setview", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "stream:waveai" [responsestream]
|
// command "streamtest" [responsestream]
|
||||||
RespStreamWaveAi(data: OpenAiStreamRequest, opts?: WshRpcCommandOpts): AsyncGenerator<OpenAIPacketType, void, boolean> {
|
StreamTestCommand(opts?: WshRpcCommandOpts): AsyncGenerator<number, void, boolean> {
|
||||||
return WOS.wshServerRpcHelper_responsestream("stream:waveai", data, opts);
|
return WOS.wshServerRpcHelper_responsestream("streamtest", null, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "streamtest" [responsestream]
|
// command "streamwaveai" [responsestream]
|
||||||
RespStreamTest(opts?: WshRpcCommandOpts): AsyncGenerator<number, void, boolean> {
|
StreamWaveAiCommand(data: OpenAiStreamRequest, opts?: WshRpcCommandOpts): AsyncGenerator<OpenAIPacketType, void, boolean> {
|
||||||
return WOS.wshServerRpcHelper_responsestream("streamtest", null, opts);
|
return WOS.wshServerRpcHelper_responsestream("streamwaveai", data, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
--title-font-size: 18px;
|
--title-font-size: 18px;
|
||||||
--secondary-text-color: rgb(195, 200, 194);
|
--secondary-text-color: rgb(195, 200, 194);
|
||||||
--grey-text-color: #666;
|
--grey-text-color: #666;
|
||||||
--main-bg-color: #454444;
|
--main-bg-color: rgb(34, 34, 34);
|
||||||
--border-color: #333333;
|
--border-color: #333333;
|
||||||
--base-font: normal 14px / normal "Inter", sans-serif;
|
--base-font: normal 14px / normal "Inter", sans-serif;
|
||||||
--fixed-font: normal 12px / normal "Hack", monospace;
|
--fixed-font: normal 12px / normal "Hack", monospace;
|
||||||
@ -19,7 +19,7 @@
|
|||||||
--warning-color: rgb(224, 185, 86);
|
--warning-color: rgb(224, 185, 86);
|
||||||
--success-color: rgb(78, 154, 6);
|
--success-color: rgb(78, 154, 6);
|
||||||
--hover-bg-color: rgba(255, 255, 255, 0.1);
|
--hover-bg-color: rgba(255, 255, 255, 0.1);
|
||||||
--block-bg-color: rgba(255, 255, 255, 0.05);
|
--block-bg-color: rgba(0, 0, 0, 0.5);
|
||||||
|
|
||||||
/* scrollbar colors */
|
/* scrollbar colors */
|
||||||
--scrollbar-background-color: transparent;
|
--scrollbar-background-color: transparent;
|
||||||
|
@ -214,7 +214,7 @@ const TerminalView = ({ blockId, model }: TerminalViewProps) => {
|
|||||||
}
|
}
|
||||||
if (shellProcStatusRef.current != "running" && keyutil.checkKeyPressed(waveEvent, "Enter")) {
|
if (shellProcStatusRef.current != "running" && keyutil.checkKeyPressed(waveEvent, "Enter")) {
|
||||||
// restart
|
// restart
|
||||||
WshServer.BlockRestartCommand({ blockid: blockId });
|
WshServer.ControllerRestartCommand({ blockid: blockId });
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -263,7 +263,7 @@ const TerminalView = ({ blockId, model }: TerminalViewProps) => {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
const b64data = btoa(asciiVal);
|
const b64data = btoa(asciiVal);
|
||||||
WshServer.BlockInputCommand({ blockid: blockId, inputdata64: b64data });
|
WshServer.ControllerInputCommand({ blockid: blockId, inputdata64: b64data });
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -76,7 +76,7 @@ export class TermWrap {
|
|||||||
|
|
||||||
handleTermData(data: string) {
|
handleTermData(data: string) {
|
||||||
const b64data = btoa(data);
|
const b64data = btoa(data);
|
||||||
WshServer.BlockInputCommand({ blockid: this.blockId, inputdata64: b64data });
|
WshServer.ControllerInputCommand({ blockid: this.blockId, inputdata64: b64data });
|
||||||
}
|
}
|
||||||
|
|
||||||
addFocusListener(focusFn: () => void) {
|
addFocusListener(focusFn: () => void) {
|
||||||
|
@ -137,7 +137,7 @@ export class WaveAiModel implements ViewModel {
|
|||||||
opts: opts,
|
opts: opts,
|
||||||
prompt: prompt,
|
prompt: prompt,
|
||||||
};
|
};
|
||||||
const aiGen = WshServer.RespStreamWaveAi(beMsg);
|
const aiGen = WshServer.StreamWaveAiCommand(beMsg);
|
||||||
let temp = async () => {
|
let temp = async () => {
|
||||||
let fullMsg = "";
|
let fullMsg = "";
|
||||||
for await (const msg of aiGen) {
|
for await (const msg of aiGen) {
|
||||||
|
33
frontend/types/gotypes.d.ts
vendored
33
frontend/types/gotypes.d.ts
vendored
@ -187,7 +187,7 @@ declare global {
|
|||||||
// waveobj.ORef
|
// waveobj.ORef
|
||||||
type ORef = string;
|
type ORef = string;
|
||||||
|
|
||||||
// waveai.OpenAIOptsType
|
// wshrpc.OpenAIOptsType
|
||||||
type OpenAIOptsType = {
|
type OpenAIOptsType = {
|
||||||
model: string;
|
model: string;
|
||||||
apitoken: string;
|
apitoken: string;
|
||||||
@ -197,7 +197,7 @@ declare global {
|
|||||||
timeout?: number;
|
timeout?: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
// waveai.OpenAIPacketType
|
// wshrpc.OpenAIPacketType
|
||||||
type OpenAIPacketType = {
|
type OpenAIPacketType = {
|
||||||
type: string;
|
type: string;
|
||||||
model?: string;
|
model?: string;
|
||||||
@ -209,21 +209,21 @@ declare global {
|
|||||||
error?: string;
|
error?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
// waveai.OpenAIPromptMessageType
|
// wshrpc.OpenAIPromptMessageType
|
||||||
type OpenAIPromptMessageType = {
|
type OpenAIPromptMessageType = {
|
||||||
role: string;
|
role: string;
|
||||||
content: string;
|
content: string;
|
||||||
name?: string;
|
name?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
// waveai.OpenAIUsageType
|
// wshrpc.OpenAIUsageType
|
||||||
type OpenAIUsageType = {
|
type OpenAIUsageType = {
|
||||||
prompt_tokens?: number;
|
prompt_tokens?: number;
|
||||||
completion_tokens?: number;
|
completion_tokens?: number;
|
||||||
total_tokens?: number;
|
total_tokens?: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
// waveai.OpenAiStreamRequest
|
// wshrpc.OpenAiStreamRequest
|
||||||
type OpenAiStreamRequest = {
|
type OpenAiStreamRequest = {
|
||||||
clientid?: string;
|
clientid?: string;
|
||||||
opts: OpenAIOptsType;
|
opts: OpenAIOptsType;
|
||||||
@ -270,6 +270,7 @@ declare global {
|
|||||||
blockheader: BlockHeaderOpts;
|
blockheader: BlockHeaderOpts;
|
||||||
autoupdate: AutoUpdateOpts;
|
autoupdate: AutoUpdateOpts;
|
||||||
termthemes: {[key: string]: TermThemeType};
|
termthemes: {[key: string]: TermThemeType};
|
||||||
|
window: WindowSettingsType;
|
||||||
};
|
};
|
||||||
|
|
||||||
// wstore.StickerClickOptsType
|
// wstore.StickerClickOptsType
|
||||||
@ -293,6 +294,13 @@ declare global {
|
|||||||
display: StickerDisplayOptsType;
|
display: StickerDisplayOptsType;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// wshrpc.SubscriptionRequest
|
||||||
|
type SubscriptionRequest = {
|
||||||
|
event: string;
|
||||||
|
scopes?: string[];
|
||||||
|
allscopes?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
// wstore.Tab
|
// wstore.Tab
|
||||||
type Tab = WaveObj & {
|
type Tab = WaveObj & {
|
||||||
name: string;
|
name: string;
|
||||||
@ -428,6 +436,14 @@ declare global {
|
|||||||
error: string;
|
error: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// wshrpc.WaveEvent
|
||||||
|
type WaveEvent = {
|
||||||
|
event: string;
|
||||||
|
scopes?: string[];
|
||||||
|
sender?: string;
|
||||||
|
data?: any;
|
||||||
|
};
|
||||||
|
|
||||||
// filestore.WaveFile
|
// filestore.WaveFile
|
||||||
type WaveFile = {
|
type WaveFile = {
|
||||||
zoneid: string;
|
zoneid: string;
|
||||||
@ -497,6 +513,13 @@ declare global {
|
|||||||
height: number;
|
height: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// wconfig.WindowSettingsType
|
||||||
|
type WindowSettingsType = {
|
||||||
|
transparent: boolean;
|
||||||
|
opacity: number;
|
||||||
|
bgcolor: string;
|
||||||
|
};
|
||||||
|
|
||||||
// wstore.Workspace
|
// wstore.Workspace
|
||||||
type Workspace = WaveObj & {
|
type Workspace = WaveObj & {
|
||||||
name: string;
|
name: string;
|
||||||
|
@ -34,6 +34,10 @@ function base64ToArray(b64: string): Uint8Array {
|
|||||||
return rtnArr;
|
return rtnArr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function boundNumber(num: number, min: number, max: number): number {
|
||||||
|
return Math.min(Math.max(num, min), max);
|
||||||
|
}
|
||||||
|
|
||||||
// works for json-like objects (arrays, objects, strings, numbers, booleans)
|
// works for json-like objects (arrays, objects, strings, numbers, booleans)
|
||||||
function jsonDeepEqual(v1: any, v2: any): boolean {
|
function jsonDeepEqual(v1: any, v2: any): boolean {
|
||||||
if (v1 === v2) {
|
if (v1 === v2) {
|
||||||
@ -193,6 +197,7 @@ function getCrypto() {
|
|||||||
export {
|
export {
|
||||||
base64ToArray,
|
base64ToArray,
|
||||||
base64ToString,
|
base64ToString,
|
||||||
|
boundNumber,
|
||||||
fireAndForget,
|
fireAndForget,
|
||||||
getCrypto,
|
getCrypto,
|
||||||
getPromiseState,
|
getPromiseState,
|
||||||
|
@ -27,6 +27,7 @@ loadFonts();
|
|||||||
(window as any).globalWS = globalWS;
|
(window as any).globalWS = globalWS;
|
||||||
(window as any).WOS = WOS;
|
(window as any).WOS = WOS;
|
||||||
(window as any).globalStore = globalStore;
|
(window as any).globalStore = globalStore;
|
||||||
|
(window as any).globalAtoms = atoms;
|
||||||
(window as any).WshServer = WshServer;
|
(window as any).WshServer = WshServer;
|
||||||
(window as any).isFullScreen = false;
|
(window as any).isFullScreen = false;
|
||||||
|
|
||||||
|
@ -76,11 +76,13 @@
|
|||||||
"@table-nav/core": "^0.0.7",
|
"@table-nav/core": "^0.0.7",
|
||||||
"@table-nav/react": "^0.0.7",
|
"@table-nav/react": "^0.0.7",
|
||||||
"@tanstack/react-table": "^8.19.3",
|
"@tanstack/react-table": "^8.19.3",
|
||||||
|
"@types/color": "^3.0.6",
|
||||||
"@xterm/addon-fit": "^0.10.0",
|
"@xterm/addon-fit": "^0.10.0",
|
||||||
"@xterm/addon-serialize": "^0.13.0",
|
"@xterm/addon-serialize": "^0.13.0",
|
||||||
"@xterm/xterm": "^5.5.0",
|
"@xterm/xterm": "^5.5.0",
|
||||||
"base64-js": "^1.5.1",
|
"base64-js": "^1.5.1",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
|
"color": "^4.2.3",
|
||||||
"dayjs": "^1.11.12",
|
"dayjs": "^1.11.12",
|
||||||
"electron-updater": "6.3.1",
|
"electron-updater": "6.3.1",
|
||||||
"html-to-image": "^1.11.11",
|
"html-to-image": "^1.11.11",
|
||||||
|
@ -6,7 +6,9 @@ package blockcontroller
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
|
"crypto/rand"
|
||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
|
"encoding/hex"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
@ -21,12 +23,13 @@ import (
|
|||||||
"github.com/wavetermdev/thenextwave/pkg/shellexec"
|
"github.com/wavetermdev/thenextwave/pkg/shellexec"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wavebase"
|
"github.com/wavetermdev/thenextwave/pkg/wavebase"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
||||||
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
||||||
)
|
)
|
||||||
|
|
||||||
// set by main-server.go (for dependency inversion)
|
// set by main-server.go (for dependency inversion)
|
||||||
var WshServerFactoryFn func(inputCh chan []byte, outputCh chan []byte, initialCtx wshutil.RpcContext) = nil
|
var WshServerFactoryFn func(inputCh chan []byte, outputCh chan []byte, initialCtx wshrpc.RpcContext) = nil
|
||||||
|
|
||||||
const (
|
const (
|
||||||
BlockController_Shell = "shell"
|
BlockController_Shell = "shell"
|
||||||
@ -205,6 +208,46 @@ func (bc *BlockController) resetTerminalState() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func getMetaBool(meta map[string]any, key string, def bool) bool {
|
||||||
|
val, found := meta[key]
|
||||||
|
if !found {
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
if val == nil {
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
if bval, ok := val.(bool); ok {
|
||||||
|
return bval
|
||||||
|
}
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
|
||||||
|
func getMetaStr(meta map[string]any, key string, def string) string {
|
||||||
|
val, found := meta[key]
|
||||||
|
if !found {
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
if val == nil {
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
if sval, ok := val.(string); ok {
|
||||||
|
return sval
|
||||||
|
}
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
|
||||||
|
// every byte is 4-bits of randomness
|
||||||
|
func randomHexString(numHexDigits int) (string, error) {
|
||||||
|
numBytes := (numHexDigits + 1) / 2 // Calculate the number of bytes needed
|
||||||
|
bytes := make([]byte, numBytes)
|
||||||
|
if _, err := rand.Read(bytes); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
hexStr := hex.EncodeToString(bytes)
|
||||||
|
return hexStr[:numHexDigits], nil // Return the exact number of hex digits
|
||||||
|
}
|
||||||
|
|
||||||
func (bc *BlockController) DoRunShellCommand(rc *RunShellOpts, blockMeta map[string]any) error {
|
func (bc *BlockController) DoRunShellCommand(rc *RunShellOpts, blockMeta map[string]any) error {
|
||||||
// create a circular blockfile for the output
|
// create a circular blockfile for the output
|
||||||
ctx, cancelFn := context.WithTimeout(context.Background(), 2*time.Second)
|
ctx, cancelFn := context.WithTimeout(context.Background(), 2*time.Second)
|
||||||
@ -232,12 +275,35 @@ func (bc *BlockController) DoRunShellCommand(rc *RunShellOpts, blockMeta map[str
|
|||||||
if shellProcErr != nil {
|
if shellProcErr != nil {
|
||||||
return shellProcErr
|
return shellProcErr
|
||||||
}
|
}
|
||||||
|
var remoteDomainSocketName string
|
||||||
|
remoteName := getMetaStr(blockMeta, "connection", "")
|
||||||
|
isRemote := remoteName != ""
|
||||||
|
if isRemote {
|
||||||
|
randStr, err := randomHexString(16) // 64-bits of randomness
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error generating random string: %w", err)
|
||||||
|
}
|
||||||
|
remoteDomainSocketName = fmt.Sprintf("/tmp/waveterm-%s.sock", randStr)
|
||||||
|
}
|
||||||
var cmdStr string
|
var cmdStr string
|
||||||
cmdOpts := shellexec.CommandOptsType{
|
cmdOpts := shellexec.CommandOptsType{
|
||||||
Env: make(map[string]string),
|
Env: make(map[string]string),
|
||||||
}
|
}
|
||||||
// temporary for blockid (will switch to a JWT at some point)
|
if !getMetaBool(blockMeta, "nowsh", false) {
|
||||||
cmdOpts.Env["LC_WAVETERM_BLOCKID"] = bc.BlockId
|
if isRemote {
|
||||||
|
jwtStr, err := wshutil.MakeClientJWTToken(wshrpc.RpcContext{TabId: bc.TabId, BlockId: bc.BlockId}, remoteDomainSocketName)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error making jwt token: %w", err)
|
||||||
|
}
|
||||||
|
cmdOpts.Env["WAVETERM_JWT"] = jwtStr
|
||||||
|
} else {
|
||||||
|
jwtStr, err := wshutil.MakeClientJWTToken(wshrpc.RpcContext{TabId: bc.TabId, BlockId: bc.BlockId}, wavebase.GetDomainSocketName())
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error making jwt token: %w", err)
|
||||||
|
}
|
||||||
|
cmdOpts.Env["WAVETERM_JWT"] = jwtStr
|
||||||
|
}
|
||||||
|
}
|
||||||
if bc.ControllerType == BlockController_Shell {
|
if bc.ControllerType == BlockController_Shell {
|
||||||
cmdOpts.Interactive = true
|
cmdOpts.Interactive = true
|
||||||
cmdOpts.Login = true
|
cmdOpts.Login = true
|
||||||
@ -284,11 +350,8 @@ func (bc *BlockController) DoRunShellCommand(rc *RunShellOpts, blockMeta map[str
|
|||||||
} else {
|
} else {
|
||||||
return fmt.Errorf("unknown controller type %q", bc.ControllerType)
|
return fmt.Errorf("unknown controller type %q", bc.ControllerType)
|
||||||
}
|
}
|
||||||
// pty buffer equivalent for ssh? i think if i have the ecmd or session i can manage it with output
|
|
||||||
// pty write needs stdin, so if i provide that, i might be able to write that way
|
|
||||||
// need a way to handle setsize???
|
|
||||||
var shellProc *shellexec.ShellProc
|
var shellProc *shellexec.ShellProc
|
||||||
if remoteName, ok := blockMeta["connection"].(string); ok && remoteName != "" {
|
if remoteName != "" {
|
||||||
shellProc, err = shellexec.StartRemoteShellProc(rc.TermSize, cmdStr, cmdOpts, remoteName)
|
shellProc, err = shellexec.StartRemoteShellProc(rc.TermSize, cmdStr, cmdOpts, remoteName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@ -309,7 +372,7 @@ func (bc *BlockController) DoRunShellCommand(rc *RunShellOpts, blockMeta map[str
|
|||||||
messageCh := make(chan []byte, 32)
|
messageCh := make(chan []byte, 32)
|
||||||
ptyBuffer := wshutil.MakePtyBuffer(wshutil.WaveOSCPrefix, bc.ShellProc.Pty, messageCh)
|
ptyBuffer := wshutil.MakePtyBuffer(wshutil.WaveOSCPrefix, bc.ShellProc.Pty, messageCh)
|
||||||
outputCh := make(chan []byte, 32)
|
outputCh := make(chan []byte, 32)
|
||||||
WshServerFactoryFn(messageCh, outputCh, wshutil.RpcContext{BlockId: bc.BlockId, TabId: bc.TabId})
|
WshServerFactoryFn(messageCh, outputCh, wshrpc.RpcContext{BlockId: bc.BlockId, TabId: bc.TabId})
|
||||||
go func() {
|
go func() {
|
||||||
// handles regular output from the pty (goes to the blockfile and xterm)
|
// handles regular output from the pty (goes to the blockfile and xterm)
|
||||||
defer func() {
|
defer func() {
|
||||||
|
@ -20,7 +20,6 @@ import (
|
|||||||
"github.com/wavetermdev/thenextwave/pkg/wconfig"
|
"github.com/wavetermdev/thenextwave/pkg/wconfig"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/web/webcmd"
|
"github.com/wavetermdev/thenextwave/pkg/web/webcmd"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshrpc/wshserver"
|
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
||||||
)
|
)
|
||||||
@ -61,10 +60,13 @@ var uiContextRType = reflect.TypeOf((*wstore.UIContext)(nil)).Elem()
|
|||||||
var waveObjRType = reflect.TypeOf((*waveobj.WaveObj)(nil)).Elem()
|
var waveObjRType = reflect.TypeOf((*waveobj.WaveObj)(nil)).Elem()
|
||||||
var updatesRtnRType = reflect.TypeOf(wstore.UpdatesRtnType{})
|
var updatesRtnRType = reflect.TypeOf(wstore.UpdatesRtnType{})
|
||||||
var orefRType = reflect.TypeOf((*waveobj.ORef)(nil)).Elem()
|
var orefRType = reflect.TypeOf((*waveobj.ORef)(nil)).Elem()
|
||||||
|
var wshRpcInterfaceRType = reflect.TypeOf((*wshrpc.WshRpcInterface)(nil)).Elem()
|
||||||
|
|
||||||
func generateTSMethodTypes(method reflect.Method, tsTypesMap map[reflect.Type]string) error {
|
func generateTSMethodTypes(method reflect.Method, tsTypesMap map[reflect.Type]string, skipFirstArg bool) error {
|
||||||
for idx := 1; idx < method.Type.NumIn(); idx++ {
|
for idx := 0; idx < method.Type.NumIn(); idx++ {
|
||||||
// skip receiver
|
if skipFirstArg && idx == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
inType := method.Type.In(idx)
|
inType := method.Type.In(idx)
|
||||||
GenerateTSType(inType, tsTypesMap)
|
GenerateTSType(inType, tsTypesMap)
|
||||||
}
|
}
|
||||||
@ -159,14 +161,13 @@ var tsRenameMap = map[string]string{
|
|||||||
|
|
||||||
func generateTSTypeInternal(rtype reflect.Type, tsTypesMap map[reflect.Type]string) (string, []reflect.Type) {
|
func generateTSTypeInternal(rtype reflect.Type, tsTypesMap map[reflect.Type]string) (string, []reflect.Type) {
|
||||||
var buf bytes.Buffer
|
var buf bytes.Buffer
|
||||||
waveObjType := reflect.TypeOf((*waveobj.WaveObj)(nil)).Elem()
|
|
||||||
tsTypeName := rtype.Name()
|
tsTypeName := rtype.Name()
|
||||||
if tsRename, ok := tsRenameMap[tsTypeName]; ok {
|
if tsRename, ok := tsRenameMap[tsTypeName]; ok {
|
||||||
tsTypeName = tsRename
|
tsTypeName = tsRename
|
||||||
}
|
}
|
||||||
var isWaveObj bool
|
var isWaveObj bool
|
||||||
buf.WriteString(fmt.Sprintf("// %s\n", rtype.String()))
|
buf.WriteString(fmt.Sprintf("// %s\n", rtype.String()))
|
||||||
if rtype.Implements(waveObjType) || reflect.PointerTo(rtype).Implements(waveObjType) {
|
if rtype.Implements(waveObjRType) || reflect.PointerTo(rtype).Implements(waveObjRType) {
|
||||||
isWaveObj = true
|
isWaveObj = true
|
||||||
buf.WriteString(fmt.Sprintf("type %s = WaveObj & {\n", tsTypeName))
|
buf.WriteString(fmt.Sprintf("type %s = WaveObj & {\n", tsTypeName))
|
||||||
} else {
|
} else {
|
||||||
@ -253,6 +254,9 @@ func GenerateTSType(rtype reflect.Type, tsTypesMap map[reflect.Type]string) {
|
|||||||
if rtype == nil {
|
if rtype == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
if rtype.Kind() == reflect.Chan {
|
||||||
|
rtype = rtype.Elem()
|
||||||
|
}
|
||||||
if rtype == metaRType {
|
if rtype == metaRType {
|
||||||
tsTypesMap[metaRType] = GenerateMetaType()
|
tsTypesMap[metaRType] = GenerateMetaType()
|
||||||
return
|
return
|
||||||
@ -397,17 +401,17 @@ func GenerateServiceClass(serviceName string, serviceObj any, tsTypesMap map[ref
|
|||||||
return sb.String()
|
return sb.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
func GenerateWshServerMethod(methodDecl *wshserver.WshServerMethodDecl, tsTypesMap map[reflect.Type]string) string {
|
func GenerateWshServerMethod(methodDecl *wshrpc.WshRpcMethodDecl, tsTypesMap map[reflect.Type]string) string {
|
||||||
if methodDecl.CommandType == wshutil.RpcType_ResponseStream {
|
if methodDecl.CommandType == wshrpc.RpcType_ResponseStream {
|
||||||
return GenerateWshServerMethod_ResponseStream(methodDecl, tsTypesMap)
|
return GenerateWshServerMethod_ResponseStream(methodDecl, tsTypesMap)
|
||||||
} else if methodDecl.CommandType == wshutil.RpcType_Call {
|
} else if methodDecl.CommandType == wshrpc.RpcType_Call {
|
||||||
return GenerateWshServerMethod_Call(methodDecl, tsTypesMap)
|
return GenerateWshServerMethod_Call(methodDecl, tsTypesMap)
|
||||||
} else {
|
} else {
|
||||||
panic(fmt.Sprintf("cannot generate wshserver commandtype %q", methodDecl.CommandType))
|
panic(fmt.Sprintf("cannot generate wshserver commandtype %q", methodDecl.CommandType))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func GenerateWshServerMethod_ResponseStream(methodDecl *wshserver.WshServerMethodDecl, tsTypesMap map[reflect.Type]string) string {
|
func GenerateWshServerMethod_ResponseStream(methodDecl *wshrpc.WshRpcMethodDecl, tsTypesMap map[reflect.Type]string) string {
|
||||||
var sb strings.Builder
|
var sb strings.Builder
|
||||||
sb.WriteString(fmt.Sprintf(" // command %q [%s]\n", methodDecl.Command, methodDecl.CommandType))
|
sb.WriteString(fmt.Sprintf(" // command %q [%s]\n", methodDecl.Command, methodDecl.CommandType))
|
||||||
respType := "any"
|
respType := "any"
|
||||||
@ -429,7 +433,7 @@ func GenerateWshServerMethod_ResponseStream(methodDecl *wshserver.WshServerMetho
|
|||||||
return sb.String()
|
return sb.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
func GenerateWshServerMethod_Call(methodDecl *wshserver.WshServerMethodDecl, tsTypesMap map[reflect.Type]string) string {
|
func GenerateWshServerMethod_Call(methodDecl *wshrpc.WshRpcMethodDecl, tsTypesMap map[reflect.Type]string) string {
|
||||||
var sb strings.Builder
|
var sb strings.Builder
|
||||||
sb.WriteString(fmt.Sprintf(" // command %q [%s]\n", methodDecl.Command, methodDecl.CommandType))
|
sb.WriteString(fmt.Sprintf(" // command %q [%s]\n", methodDecl.Command, methodDecl.CommandType))
|
||||||
rtnType := "Promise<void>"
|
rtnType := "Promise<void>"
|
||||||
@ -469,7 +473,7 @@ func GenerateServiceTypes(tsTypesMap map[reflect.Type]string) error {
|
|||||||
serviceType := reflect.TypeOf(serviceObj)
|
serviceType := reflect.TypeOf(serviceObj)
|
||||||
for midx := 0; midx < serviceType.NumMethod(); midx++ {
|
for midx := 0; midx < serviceType.NumMethod(); midx++ {
|
||||||
method := serviceType.Method(midx)
|
method := serviceType.Method(midx)
|
||||||
err := generateTSMethodTypes(method, tsTypesMap)
|
err := generateTSMethodTypes(method, tsTypesMap, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("error generating TS method types for %s.%s: %v", serviceType, method.Name, err)
|
return fmt.Errorf("error generating TS method types for %s.%s: %v", serviceType, method.Name, err)
|
||||||
}
|
}
|
||||||
@ -480,16 +484,12 @@ func GenerateServiceTypes(tsTypesMap map[reflect.Type]string) error {
|
|||||||
|
|
||||||
func GenerateWshServerTypes(tsTypesMap map[reflect.Type]string) error {
|
func GenerateWshServerTypes(tsTypesMap map[reflect.Type]string) error {
|
||||||
GenerateTSType(reflect.TypeOf(wshrpc.WshRpcCommandOpts{}), tsTypesMap)
|
GenerateTSType(reflect.TypeOf(wshrpc.WshRpcCommandOpts{}), tsTypesMap)
|
||||||
for _, methodDecl := range wshserver.WshServerCommandToDeclMap {
|
rtype := wshRpcInterfaceRType
|
||||||
GenerateTSType(methodDecl.CommandDataType, tsTypesMap)
|
for midx := 0; midx < rtype.NumMethod(); midx++ {
|
||||||
if methodDecl.DefaultResponseDataType != nil {
|
method := rtype.Method(midx)
|
||||||
GenerateTSType(methodDecl.DefaultResponseDataType, tsTypesMap)
|
err := generateTSMethodTypes(method, tsTypesMap, false)
|
||||||
}
|
if err != nil {
|
||||||
for _, rtype := range methodDecl.RequestDataTypes {
|
return fmt.Errorf("error generating TS method types for %s.%s: %v", rtype, method.Name, err)
|
||||||
GenerateTSType(rtype, tsTypesMap)
|
|
||||||
}
|
|
||||||
for _, rtype := range methodDecl.ResponseDataTypes {
|
|
||||||
GenerateTSType(rtype, tsTypesMap)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
@ -800,3 +800,29 @@ func MoveSliceIdxToFront[T any](arr []T, idx int) []T {
|
|||||||
rtn = append(rtn, arr[idx+1:]...)
|
rtn = append(rtn, arr[idx+1:]...)
|
||||||
return rtn
|
return rtn
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// matches a delimited string with a pattern string
|
||||||
|
// the pattern string can contain "*" to match a single part, or "**" to match the rest of the string
|
||||||
|
// note that "**" may only appear at the end of the string
|
||||||
|
func StarMatchString(pattern string, s string, delimiter string) bool {
|
||||||
|
patternParts := strings.Split(pattern, delimiter)
|
||||||
|
stringParts := strings.Split(s, delimiter)
|
||||||
|
pLen, sLen := len(patternParts), len(stringParts)
|
||||||
|
|
||||||
|
for i := 0; i < pLen; i++ {
|
||||||
|
if patternParts[i] == "**" {
|
||||||
|
// '**' must be at the end to be valid
|
||||||
|
return i == pLen-1
|
||||||
|
}
|
||||||
|
if i >= sLen {
|
||||||
|
// If string is exhausted but pattern is not
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if patternParts[i] != "*" && patternParts[i] != stringParts[i] {
|
||||||
|
// If current parts don't match and pattern part is not '*'
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Check if both pattern and string are fully matched
|
||||||
|
return pLen == sLen
|
||||||
|
}
|
||||||
|
@ -23,12 +23,6 @@ const OpenAIPacketStr = "openai"
|
|||||||
const OpenAICloudReqStr = "openai-cloudreq"
|
const OpenAICloudReqStr = "openai-cloudreq"
|
||||||
const PacketEOFStr = "EOF"
|
const PacketEOFStr = "EOF"
|
||||||
|
|
||||||
type OpenAIUsageType struct {
|
|
||||||
PromptTokens int `json:"prompt_tokens,omitempty"`
|
|
||||||
CompletionTokens int `json:"completion_tokens,omitempty"`
|
|
||||||
TotalTokens int `json:"total_tokens,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type OpenAICmdInfoPacketOutputType struct {
|
type OpenAICmdInfoPacketOutputType struct {
|
||||||
Model string `json:"model,omitempty"`
|
Model string `json:"model,omitempty"`
|
||||||
Created int64 `json:"created,omitempty"`
|
Created int64 `json:"created,omitempty"`
|
||||||
@ -37,19 +31,8 @@ type OpenAICmdInfoPacketOutputType struct {
|
|||||||
Error string `json:"error,omitempty"`
|
Error string `json:"error,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type OpenAIPacketType struct {
|
func MakeOpenAIPacket() *wshrpc.OpenAIPacketType {
|
||||||
Type string `json:"type"`
|
return &wshrpc.OpenAIPacketType{Type: OpenAIPacketStr}
|
||||||
Model string `json:"model,omitempty"`
|
|
||||||
Created int64 `json:"created,omitempty"`
|
|
||||||
FinishReason string `json:"finish_reason,omitempty"`
|
|
||||||
Usage *OpenAIUsageType `json:"usage,omitempty"`
|
|
||||||
Index int `json:"index,omitempty"`
|
|
||||||
Text string `json:"text,omitempty"`
|
|
||||||
Error string `json:"error,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func MakeOpenAIPacket() *OpenAIPacketType {
|
|
||||||
return &OpenAIPacketType{Type: OpenAIPacketStr}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type OpenAICmdInfoChatMessage struct {
|
type OpenAICmdInfoChatMessage struct {
|
||||||
@ -60,41 +43,20 @@ type OpenAICmdInfoChatMessage struct {
|
|||||||
UserEngineeredQuery string `json:"userengineeredquery,omitempty"`
|
UserEngineeredQuery string `json:"userengineeredquery,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type OpenAIPromptMessageType struct {
|
|
||||||
Role string `json:"role"`
|
|
||||||
Content string `json:"content"`
|
|
||||||
Name string `json:"name,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type OpenAICloudReqPacketType struct {
|
type OpenAICloudReqPacketType struct {
|
||||||
Type string `json:"type"`
|
Type string `json:"type"`
|
||||||
ClientId string `json:"clientid"`
|
ClientId string `json:"clientid"`
|
||||||
Prompt []OpenAIPromptMessageType `json:"prompt"`
|
Prompt []wshrpc.OpenAIPromptMessageType `json:"prompt"`
|
||||||
MaxTokens int `json:"maxtokens,omitempty"`
|
MaxTokens int `json:"maxtokens,omitempty"`
|
||||||
MaxChoices int `json:"maxchoices,omitempty"`
|
MaxChoices int `json:"maxchoices,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type OpenAIOptsType struct {
|
|
||||||
Model string `json:"model"`
|
|
||||||
APIToken string `json:"apitoken"`
|
|
||||||
BaseURL string `json:"baseurl,omitempty"`
|
|
||||||
MaxTokens int `json:"maxtokens,omitempty"`
|
|
||||||
MaxChoices int `json:"maxchoices,omitempty"`
|
|
||||||
Timeout int `json:"timeout,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func MakeOpenAICloudReqPacket() *OpenAICloudReqPacketType {
|
func MakeOpenAICloudReqPacket() *OpenAICloudReqPacketType {
|
||||||
return &OpenAICloudReqPacketType{
|
return &OpenAICloudReqPacketType{
|
||||||
Type: OpenAICloudReqStr,
|
Type: OpenAICloudReqStr,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type OpenAiStreamRequest struct {
|
|
||||||
ClientId string `json:"clientid,omitempty"`
|
|
||||||
Opts *OpenAIOptsType `json:"opts"`
|
|
||||||
Prompt []OpenAIPromptMessageType `json:"prompt"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetWSEndpoint() string {
|
func GetWSEndpoint() string {
|
||||||
return PCloudWSEndpoint
|
return PCloudWSEndpoint
|
||||||
if !wavebase.IsDevMode() {
|
if !wavebase.IsDevMode() {
|
||||||
@ -116,18 +78,18 @@ const PCloudWSEndpointVarName = "PCLOUD_WS_ENDPOINT"
|
|||||||
|
|
||||||
const CloudWebsocketConnectTimeout = 1 * time.Minute
|
const CloudWebsocketConnectTimeout = 1 * time.Minute
|
||||||
|
|
||||||
func convertUsage(resp openaiapi.ChatCompletionResponse) *OpenAIUsageType {
|
func convertUsage(resp openaiapi.ChatCompletionResponse) *wshrpc.OpenAIUsageType {
|
||||||
if resp.Usage.TotalTokens == 0 {
|
if resp.Usage.TotalTokens == 0 {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return &OpenAIUsageType{
|
return &wshrpc.OpenAIUsageType{
|
||||||
PromptTokens: resp.Usage.PromptTokens,
|
PromptTokens: resp.Usage.PromptTokens,
|
||||||
CompletionTokens: resp.Usage.CompletionTokens,
|
CompletionTokens: resp.Usage.CompletionTokens,
|
||||||
TotalTokens: resp.Usage.TotalTokens,
|
TotalTokens: resp.Usage.TotalTokens,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func ConvertPrompt(prompt []OpenAIPromptMessageType) []openaiapi.ChatCompletionMessage {
|
func ConvertPrompt(prompt []wshrpc.OpenAIPromptMessageType) []openaiapi.ChatCompletionMessage {
|
||||||
var rtn []openaiapi.ChatCompletionMessage
|
var rtn []openaiapi.ChatCompletionMessage
|
||||||
for _, p := range prompt {
|
for _, p := range prompt {
|
||||||
msg := openaiapi.ChatCompletionMessage{Role: p.Role, Content: p.Content, Name: p.Name}
|
msg := openaiapi.ChatCompletionMessage{Role: p.Role, Content: p.Content, Name: p.Name}
|
||||||
@ -136,31 +98,31 @@ func ConvertPrompt(prompt []OpenAIPromptMessageType) []openaiapi.ChatCompletionM
|
|||||||
return rtn
|
return rtn
|
||||||
}
|
}
|
||||||
|
|
||||||
func RunCloudCompletionStream(ctx context.Context, request OpenAiStreamRequest) chan wshrpc.RespOrErrorUnion[OpenAIPacketType] {
|
func RunCloudCompletionStream(ctx context.Context, request wshrpc.OpenAiStreamRequest) chan wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType] {
|
||||||
rtn := make(chan wshrpc.RespOrErrorUnion[OpenAIPacketType])
|
rtn := make(chan wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType])
|
||||||
go func() {
|
go func() {
|
||||||
log.Printf("start: %v", request)
|
log.Printf("start: %v", request)
|
||||||
defer close(rtn)
|
defer close(rtn)
|
||||||
if request.Opts == nil {
|
if request.Opts == nil {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("no openai opts found")}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("no openai opts found")}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
websocketContext, dialCancelFn := context.WithTimeout(context.Background(), CloudWebsocketConnectTimeout)
|
websocketContext, dialCancelFn := context.WithTimeout(context.Background(), CloudWebsocketConnectTimeout)
|
||||||
defer dialCancelFn()
|
defer dialCancelFn()
|
||||||
conn, _, err := websocket.DefaultDialer.DialContext(websocketContext, GetWSEndpoint(), nil)
|
conn, _, err := websocket.DefaultDialer.DialContext(websocketContext, GetWSEndpoint(), nil)
|
||||||
|
if err == context.DeadlineExceeded {
|
||||||
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, timed out connecting to cloud server: %v", err)}
|
||||||
|
return
|
||||||
|
} else if err != nil {
|
||||||
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket connect error: %v", err)}
|
||||||
|
return
|
||||||
|
}
|
||||||
defer func() {
|
defer func() {
|
||||||
err = conn.Close()
|
err = conn.Close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("unable to close openai channel: %v", err)}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("unable to close openai channel: %v", err)}
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
if err == context.DeadlineExceeded {
|
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, timed out connecting to cloud server: %v", err)}
|
|
||||||
return
|
|
||||||
} else if err != nil {
|
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket connect error: %v", err)}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
reqPk := MakeOpenAICloudReqPacket()
|
reqPk := MakeOpenAICloudReqPacket()
|
||||||
reqPk.ClientId = request.ClientId
|
reqPk.ClientId = request.ClientId
|
||||||
reqPk.Prompt = request.Prompt
|
reqPk.Prompt = request.Prompt
|
||||||
@ -168,12 +130,12 @@ func RunCloudCompletionStream(ctx context.Context, request OpenAiStreamRequest)
|
|||||||
reqPk.MaxChoices = request.Opts.MaxChoices
|
reqPk.MaxChoices = request.Opts.MaxChoices
|
||||||
configMessageBuf, err := json.Marshal(reqPk)
|
configMessageBuf, err := json.Marshal(reqPk)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, packet marshal error: %v", err)}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, packet marshal error: %v", err)}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
err = conn.WriteMessage(websocket.TextMessage, configMessageBuf)
|
err = conn.WriteMessage(websocket.TextMessage, configMessageBuf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket write config error: %v", err)}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket write config error: %v", err)}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
for {
|
for {
|
||||||
@ -184,14 +146,14 @@ func RunCloudCompletionStream(ctx context.Context, request OpenAiStreamRequest)
|
|||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("err received: %v", err)
|
log.Printf("err received: %v", err)
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket error reading message: %v", err)}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket error reading message: %v", err)}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
var streamResp *OpenAIPacketType
|
var streamResp *wshrpc.OpenAIPacketType
|
||||||
err = json.Unmarshal(socketMessage, &streamResp)
|
err = json.Unmarshal(socketMessage, &streamResp)
|
||||||
log.Printf("ai resp: %v", streamResp)
|
log.Printf("ai resp: %v", streamResp)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket response json decode error: %v", err)}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket response json decode error: %v", err)}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
if streamResp.Error == PacketEOFStr {
|
if streamResp.Error == PacketEOFStr {
|
||||||
@ -199,30 +161,30 @@ func RunCloudCompletionStream(ctx context.Context, request OpenAiStreamRequest)
|
|||||||
break
|
break
|
||||||
} else if streamResp.Error != "" {
|
} else if streamResp.Error != "" {
|
||||||
// use error from server directly
|
// use error from server directly
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("%v", streamResp.Error)}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("%v", streamResp.Error)}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Response: *streamResp}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Response: *streamResp}
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
return rtn
|
return rtn
|
||||||
}
|
}
|
||||||
|
|
||||||
func RunLocalCompletionStream(ctx context.Context, request OpenAiStreamRequest) chan wshrpc.RespOrErrorUnion[OpenAIPacketType] {
|
func RunLocalCompletionStream(ctx context.Context, request wshrpc.OpenAiStreamRequest) chan wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType] {
|
||||||
rtn := make(chan wshrpc.RespOrErrorUnion[OpenAIPacketType])
|
rtn := make(chan wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType])
|
||||||
go func() {
|
go func() {
|
||||||
log.Printf("start2: %v", request)
|
log.Printf("start2: %v", request)
|
||||||
defer close(rtn)
|
defer close(rtn)
|
||||||
if request.Opts == nil {
|
if request.Opts == nil {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("no openai opts found")}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("no openai opts found")}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if request.Opts.Model == "" {
|
if request.Opts.Model == "" {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("no openai model specified")}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("no openai model specified")}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if request.Opts.BaseURL == "" && request.Opts.APIToken == "" {
|
if request.Opts.BaseURL == "" && request.Opts.APIToken == "" {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("no api token")}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("no api token")}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
clientConfig := openaiapi.DefaultConfig(request.Opts.APIToken)
|
clientConfig := openaiapi.DefaultConfig(request.Opts.APIToken)
|
||||||
@ -241,7 +203,7 @@ func RunLocalCompletionStream(ctx context.Context, request OpenAiStreamRequest)
|
|||||||
}
|
}
|
||||||
apiResp, err := client.CreateChatCompletionStream(ctx, req)
|
apiResp, err := client.CreateChatCompletionStream(ctx, req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("error calling openai API: %v", err)}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("error calling openai API: %v", err)}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
sentHeader := false
|
sentHeader := false
|
||||||
@ -253,14 +215,14 @@ func RunLocalCompletionStream(ctx context.Context, request OpenAiStreamRequest)
|
|||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("err received2: %v", err)
|
log.Printf("err received2: %v", err)
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket error reading message: %v", err)}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Error: fmt.Errorf("OpenAI request, websocket error reading message: %v", err)}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
if streamResp.Model != "" && !sentHeader {
|
if streamResp.Model != "" && !sentHeader {
|
||||||
pk := MakeOpenAIPacket()
|
pk := MakeOpenAIPacket()
|
||||||
pk.Model = streamResp.Model
|
pk.Model = streamResp.Model
|
||||||
pk.Created = streamResp.Created
|
pk.Created = streamResp.Created
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Response: *pk}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Response: *pk}
|
||||||
sentHeader = true
|
sentHeader = true
|
||||||
}
|
}
|
||||||
for _, choice := range streamResp.Choices {
|
for _, choice := range streamResp.Choices {
|
||||||
@ -268,15 +230,15 @@ func RunLocalCompletionStream(ctx context.Context, request OpenAiStreamRequest)
|
|||||||
pk.Index = choice.Index
|
pk.Index = choice.Index
|
||||||
pk.Text = choice.Delta.Content
|
pk.Text = choice.Delta.Content
|
||||||
pk.FinishReason = string(choice.FinishReason)
|
pk.FinishReason = string(choice.FinishReason)
|
||||||
rtn <- wshrpc.RespOrErrorUnion[OpenAIPacketType]{Response: *pk}
|
rtn <- wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType]{Response: *pk}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
return rtn
|
return rtn
|
||||||
}
|
}
|
||||||
|
|
||||||
func marshalResponse(resp openaiapi.ChatCompletionResponse) []*OpenAIPacketType {
|
func marshalResponse(resp openaiapi.ChatCompletionResponse) []*wshrpc.OpenAIPacketType {
|
||||||
var rtn []*OpenAIPacketType
|
var rtn []*wshrpc.OpenAIPacketType
|
||||||
headerPk := MakeOpenAIPacket()
|
headerPk := MakeOpenAIPacket()
|
||||||
headerPk.Model = resp.Model
|
headerPk.Model = resp.Model
|
||||||
headerPk.Created = resp.Created
|
headerPk.Created = resp.Created
|
||||||
@ -292,14 +254,14 @@ func marshalResponse(resp openaiapi.ChatCompletionResponse) []*OpenAIPacketType
|
|||||||
return rtn
|
return rtn
|
||||||
}
|
}
|
||||||
|
|
||||||
func CreateErrorPacket(errStr string) *OpenAIPacketType {
|
func CreateErrorPacket(errStr string) *wshrpc.OpenAIPacketType {
|
||||||
errPk := MakeOpenAIPacket()
|
errPk := MakeOpenAIPacket()
|
||||||
errPk.FinishReason = "error"
|
errPk.FinishReason = "error"
|
||||||
errPk.Error = errStr
|
errPk.Error = errStr
|
||||||
return errPk
|
return errPk
|
||||||
}
|
}
|
||||||
|
|
||||||
func CreateTextPacket(text string) *OpenAIPacketType {
|
func CreateTextPacket(text string) *wshrpc.OpenAIPacketType {
|
||||||
pk := MakeOpenAIPacket()
|
pk := MakeOpenAIPacket()
|
||||||
pk.Text = text
|
pk.Text = text
|
||||||
return pk
|
return pk
|
||||||
|
@ -72,6 +72,13 @@ type TermThemesConfigType map[string]TermThemeType
|
|||||||
|
|
||||||
// TODO add default term theme settings
|
// TODO add default term theme settings
|
||||||
|
|
||||||
|
// note we pointers so we preserve nulls
|
||||||
|
type WindowSettingsType struct {
|
||||||
|
Transparent *bool `json:"transparent"`
|
||||||
|
Opacity *float64 `json:"opacity"`
|
||||||
|
BgColor *string `json:"bgcolor"`
|
||||||
|
}
|
||||||
|
|
||||||
type SettingsConfigType struct {
|
type SettingsConfigType struct {
|
||||||
MimeTypes map[string]MimeTypeConfigType `json:"mimetypes"`
|
MimeTypes map[string]MimeTypeConfigType `json:"mimetypes"`
|
||||||
Term TerminalConfigType `json:"term"`
|
Term TerminalConfigType `json:"term"`
|
||||||
@ -79,6 +86,7 @@ type SettingsConfigType struct {
|
|||||||
BlockHeader BlockHeaderOpts `json:"blockheader"`
|
BlockHeader BlockHeaderOpts `json:"blockheader"`
|
||||||
AutoUpdate *AutoUpdateOpts `json:"autoupdate"`
|
AutoUpdate *AutoUpdateOpts `json:"autoupdate"`
|
||||||
TermThemes TermThemesConfigType `json:"termthemes"`
|
TermThemes TermThemesConfigType `json:"termthemes"`
|
||||||
|
WindowSettings WindowSettingsType `json:"window"`
|
||||||
}
|
}
|
||||||
|
|
||||||
var DefaultTermDarkTheme = TermThemeType{
|
var DefaultTermDarkTheme = TermThemeType{
|
||||||
|
@ -23,7 +23,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
// set by main-server.go (for dependency inversion)
|
// set by main-server.go (for dependency inversion)
|
||||||
var WshServerFactoryFn func(inputCh chan []byte, outputCh chan []byte, initialCtx wshutil.RpcContext) = nil
|
var WshServerFactoryFn func(inputCh chan []byte, outputCh chan []byte, initialCtx wshrpc.RpcContext) = nil
|
||||||
|
|
||||||
const wsReadWaitTimeout = 15 * time.Second
|
const wsReadWaitTimeout = 15 * time.Second
|
||||||
const wsWriteWaitTimeout = 10 * time.Second
|
const wsWriteWaitTimeout = 10 * time.Second
|
||||||
@ -104,7 +104,7 @@ func processWSCommand(jmsg map[string]any, outputCh chan any, rpcInputCh chan []
|
|||||||
TermSize: &cmd.TermSize,
|
TermSize: &cmd.TermSize,
|
||||||
}
|
}
|
||||||
rpcMsg := wshutil.RpcMessage{
|
rpcMsg := wshutil.RpcMessage{
|
||||||
Command: wshrpc.Command_BlockInput,
|
Command: wshrpc.Command_ControllerInput,
|
||||||
Data: data,
|
Data: data,
|
||||||
}
|
}
|
||||||
msgBytes, err := json.Marshal(rpcMsg)
|
msgBytes, err := json.Marshal(rpcMsg)
|
||||||
@ -121,7 +121,7 @@ func processWSCommand(jmsg map[string]any, outputCh chan any, rpcInputCh chan []
|
|||||||
InputData64: cmd.InputData64,
|
InputData64: cmd.InputData64,
|
||||||
}
|
}
|
||||||
rpcMsg := wshutil.RpcMessage{
|
rpcMsg := wshutil.RpcMessage{
|
||||||
Command: wshrpc.Command_BlockInput,
|
Command: wshrpc.Command_ControllerInput,
|
||||||
Data: data,
|
Data: data,
|
||||||
}
|
}
|
||||||
msgBytes, err := json.Marshal(rpcMsg)
|
msgBytes, err := json.Marshal(rpcMsg)
|
||||||
@ -281,7 +281,7 @@ func HandleWsInternal(w http.ResponseWriter, r *http.Request) error {
|
|||||||
rpcOutputCh := make(chan []byte, 32)
|
rpcOutputCh := make(chan []byte, 32)
|
||||||
eventbus.RegisterWSChannel(wsConnId, windowId, outputCh)
|
eventbus.RegisterWSChannel(wsConnId, windowId, outputCh)
|
||||||
defer eventbus.UnregisterWSChannel(wsConnId)
|
defer eventbus.UnregisterWSChannel(wsConnId)
|
||||||
WshServerFactoryFn(rpcInputCh, rpcOutputCh, wshutil.RpcContext{WindowId: windowId})
|
WshServerFactoryFn(rpcInputCh, rpcOutputCh, wshrpc.RpcContext{WindowId: windowId})
|
||||||
wg := &sync.WaitGroup{}
|
wg := &sync.WaitGroup{}
|
||||||
wg.Add(2)
|
wg.Add(2)
|
||||||
go func() {
|
go func() {
|
||||||
|
111
pkg/wps/wps.go
111
pkg/wps/wps.go
@ -5,44 +5,50 @@
|
|||||||
package wps
|
package wps
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/wavetermdev/thenextwave/pkg/util/utilfn"
|
"github.com/wavetermdev/thenextwave/pkg/util/utilfn"
|
||||||
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
)
|
)
|
||||||
|
|
||||||
// this broker interface is mostly generic
|
// this broker interface is mostly generic
|
||||||
// strong typing and event types can be defined elsewhere
|
// strong typing and event types can be defined elsewhere
|
||||||
|
|
||||||
type WaveEvent struct {
|
|
||||||
Event string `json:"event"`
|
|
||||||
Scopes []string `json:"scopes,omitempty"`
|
|
||||||
Sender string `json:"sender,omitempty"`
|
|
||||||
Data any `json:"data,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type SubscriptionRequest struct {
|
|
||||||
Event string `json:"event"`
|
|
||||||
Scopes []string `json:"scopes,omitempty"`
|
|
||||||
AllScopes bool `json:"allscopes,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Client interface {
|
type Client interface {
|
||||||
ClientId() string
|
ClientId() string
|
||||||
SendEvent(event WaveEvent)
|
SendEvent(event wshrpc.WaveEvent)
|
||||||
}
|
}
|
||||||
|
|
||||||
type BrokerSubscription struct {
|
type BrokerSubscription struct {
|
||||||
AllSubs []string // clientids of client subscribed to "all" events
|
AllSubs []string // clientids of client subscribed to "all" events
|
||||||
ScopeSubs map[string][]string // clientids of client subscribed to specific scopes
|
ScopeSubs map[string][]string // clientids of client subscribed to specific scopes
|
||||||
|
StarSubs map[string][]string // clientids of client subscribed to star scope (scopes with "*" or "**" in them)
|
||||||
}
|
}
|
||||||
|
|
||||||
type Broker struct {
|
type BrokerType struct {
|
||||||
Lock *sync.Mutex
|
Lock *sync.Mutex
|
||||||
ClientMap map[string]Client
|
ClientMap map[string]Client
|
||||||
SubMap map[string]*BrokerSubscription
|
SubMap map[string]*BrokerSubscription
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Broker) Subscribe(subscriber Client, sub SubscriptionRequest) {
|
var Broker = &BrokerType{
|
||||||
|
Lock: &sync.Mutex{},
|
||||||
|
ClientMap: make(map[string]Client),
|
||||||
|
SubMap: make(map[string]*BrokerSubscription),
|
||||||
|
}
|
||||||
|
|
||||||
|
func scopeHasStarMatch(scope string) bool {
|
||||||
|
parts := strings.Split(scope, ":")
|
||||||
|
for _, part := range parts {
|
||||||
|
if part == "*" || part == "**" {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *BrokerType) Subscribe(subscriber Client, sub wshrpc.SubscriptionRequest) {
|
||||||
b.Lock.Lock()
|
b.Lock.Lock()
|
||||||
defer b.Lock.Unlock()
|
defer b.Lock.Unlock()
|
||||||
clientId := subscriber.ClientId()
|
clientId := subscriber.ClientId()
|
||||||
@ -51,6 +57,7 @@ func (b *Broker) Subscribe(subscriber Client, sub SubscriptionRequest) {
|
|||||||
bs = &BrokerSubscription{
|
bs = &BrokerSubscription{
|
||||||
AllSubs: []string{},
|
AllSubs: []string{},
|
||||||
ScopeSubs: make(map[string][]string),
|
ScopeSubs: make(map[string][]string),
|
||||||
|
StarSubs: make(map[string][]string),
|
||||||
}
|
}
|
||||||
b.SubMap[sub.Event] = bs
|
b.SubMap[sub.Event] = bs
|
||||||
}
|
}
|
||||||
@ -58,17 +65,47 @@ func (b *Broker) Subscribe(subscriber Client, sub SubscriptionRequest) {
|
|||||||
bs.AllSubs = utilfn.AddElemToSliceUniq(bs.AllSubs, clientId)
|
bs.AllSubs = utilfn.AddElemToSliceUniq(bs.AllSubs, clientId)
|
||||||
}
|
}
|
||||||
for _, scope := range sub.Scopes {
|
for _, scope := range sub.Scopes {
|
||||||
scopeSubs := bs.ScopeSubs[scope]
|
starMatch := scopeHasStarMatch(scope)
|
||||||
scopeSubs = utilfn.AddElemToSliceUniq(scopeSubs, clientId)
|
if starMatch {
|
||||||
bs.ScopeSubs[scope] = scopeSubs
|
addStrToScopeMap(bs.StarSubs, scope, clientId)
|
||||||
|
} else {
|
||||||
|
addStrToScopeMap(bs.ScopeSubs, scope, clientId)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (bs *BrokerSubscription) IsEmpty() bool {
|
func (bs *BrokerSubscription) IsEmpty() bool {
|
||||||
return len(bs.AllSubs) == 0 && len(bs.ScopeSubs) == 0
|
return len(bs.AllSubs) == 0 && len(bs.ScopeSubs) == 0 && len(bs.StarSubs) == 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Broker) Unsubscribe(subscriber Client, sub SubscriptionRequest) {
|
func removeStrFromScopeMap(scopeMap map[string][]string, scope string, clientId string) {
|
||||||
|
scopeSubs := scopeMap[scope]
|
||||||
|
scopeSubs = utilfn.RemoveElemFromSlice(scopeSubs, clientId)
|
||||||
|
if len(scopeSubs) == 0 {
|
||||||
|
delete(scopeMap, scope)
|
||||||
|
} else {
|
||||||
|
scopeMap[scope] = scopeSubs
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func removeStrFromScopeMapAll(scopeMap map[string][]string, clientId string) {
|
||||||
|
for scope, scopeSubs := range scopeMap {
|
||||||
|
scopeSubs = utilfn.RemoveElemFromSlice(scopeSubs, clientId)
|
||||||
|
if len(scopeSubs) == 0 {
|
||||||
|
delete(scopeMap, scope)
|
||||||
|
} else {
|
||||||
|
scopeMap[scope] = scopeSubs
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func addStrToScopeMap(scopeMap map[string][]string, scope string, clientId string) {
|
||||||
|
scopeSubs := scopeMap[scope]
|
||||||
|
scopeSubs = utilfn.AddElemToSliceUniq(scopeSubs, clientId)
|
||||||
|
scopeMap[scope] = scopeSubs
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *BrokerType) Unsubscribe(subscriber Client, sub wshrpc.SubscriptionRequest) {
|
||||||
b.Lock.Lock()
|
b.Lock.Lock()
|
||||||
defer b.Lock.Unlock()
|
defer b.Lock.Unlock()
|
||||||
clientId := subscriber.ClientId()
|
clientId := subscriber.ClientId()
|
||||||
@ -80,12 +117,11 @@ func (b *Broker) Unsubscribe(subscriber Client, sub SubscriptionRequest) {
|
|||||||
bs.AllSubs = utilfn.RemoveElemFromSlice(bs.AllSubs, clientId)
|
bs.AllSubs = utilfn.RemoveElemFromSlice(bs.AllSubs, clientId)
|
||||||
}
|
}
|
||||||
for _, scope := range sub.Scopes {
|
for _, scope := range sub.Scopes {
|
||||||
scopeSubs := bs.ScopeSubs[scope]
|
starMatch := scopeHasStarMatch(scope)
|
||||||
scopeSubs = utilfn.RemoveElemFromSlice(scopeSubs, clientId)
|
if starMatch {
|
||||||
if len(scopeSubs) == 0 {
|
removeStrFromScopeMap(bs.StarSubs, scope, clientId)
|
||||||
delete(bs.ScopeSubs, scope)
|
|
||||||
} else {
|
} else {
|
||||||
bs.ScopeSubs[scope] = scopeSubs
|
removeStrFromScopeMap(bs.ScopeSubs, scope, clientId)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if bs.IsEmpty() {
|
if bs.IsEmpty() {
|
||||||
@ -93,28 +129,22 @@ func (b *Broker) Unsubscribe(subscriber Client, sub SubscriptionRequest) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Broker) UnsubscribeAll(subscriber Client) {
|
func (b *BrokerType) UnsubscribeAll(subscriber Client) {
|
||||||
b.Lock.Lock()
|
b.Lock.Lock()
|
||||||
defer b.Lock.Unlock()
|
defer b.Lock.Unlock()
|
||||||
clientId := subscriber.ClientId()
|
clientId := subscriber.ClientId()
|
||||||
delete(b.ClientMap, clientId)
|
delete(b.ClientMap, clientId)
|
||||||
for eventType, bs := range b.SubMap {
|
for eventType, bs := range b.SubMap {
|
||||||
bs.AllSubs = utilfn.RemoveElemFromSlice(bs.AllSubs, clientId)
|
bs.AllSubs = utilfn.RemoveElemFromSlice(bs.AllSubs, clientId)
|
||||||
for scope, scopeSubs := range bs.ScopeSubs {
|
removeStrFromScopeMapAll(bs.StarSubs, clientId)
|
||||||
scopeSubs = utilfn.RemoveElemFromSlice(scopeSubs, clientId)
|
removeStrFromScopeMapAll(bs.ScopeSubs, clientId)
|
||||||
if len(scopeSubs) == 0 {
|
|
||||||
delete(bs.ScopeSubs, scope)
|
|
||||||
} else {
|
|
||||||
bs.ScopeSubs[scope] = scopeSubs
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if bs.IsEmpty() {
|
if bs.IsEmpty() {
|
||||||
delete(b.SubMap, eventType)
|
delete(b.SubMap, eventType)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Broker) Publish(subscriber Client, event WaveEvent) {
|
func (b *BrokerType) Publish(event wshrpc.WaveEvent) {
|
||||||
clientIds := b.getMatchingClientIds(event)
|
clientIds := b.getMatchingClientIds(event)
|
||||||
for _, clientId := range clientIds {
|
for _, clientId := range clientIds {
|
||||||
client := b.ClientMap[clientId]
|
client := b.ClientMap[clientId]
|
||||||
@ -124,7 +154,7 @@ func (b *Broker) Publish(subscriber Client, event WaveEvent) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (b *Broker) getMatchingClientIds(event WaveEvent) []string {
|
func (b *BrokerType) getMatchingClientIds(event wshrpc.WaveEvent) []string {
|
||||||
b.Lock.Lock()
|
b.Lock.Lock()
|
||||||
defer b.Lock.Unlock()
|
defer b.Lock.Unlock()
|
||||||
bs := b.SubMap[event.Event]
|
bs := b.SubMap[event.Event]
|
||||||
@ -139,6 +169,13 @@ func (b *Broker) getMatchingClientIds(event WaveEvent) []string {
|
|||||||
for _, clientId := range bs.ScopeSubs[scope] {
|
for _, clientId := range bs.ScopeSubs[scope] {
|
||||||
clientIds[clientId] = true
|
clientIds[clientId] = true
|
||||||
}
|
}
|
||||||
|
for starScope := range bs.StarSubs {
|
||||||
|
if utilfn.StarMatchString(starScope, scope, ":") {
|
||||||
|
for _, clientId := range bs.StarSubs[starScope] {
|
||||||
|
clientIds[clientId] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
var rtn []string
|
var rtn []string
|
||||||
for clientId := range clientIds {
|
for clientId := range clientIds {
|
||||||
|
@ -9,24 +9,29 @@ import (
|
|||||||
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/waveai"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// command "controller:input", wshserver.BlockInputCommand
|
// command "authenticate", wshserver.AuthenticateCommand
|
||||||
func BlockInputCommand(w *wshutil.WshRpc, data wshrpc.CommandBlockInputData, opts *wshrpc.WshRpcCommandOpts) error {
|
func AuthenticateCommand(w *wshutil.WshRpc, data string, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
_, err := sendRpcRequestCallHelper[any](w, "controller:input", data, opts)
|
_, err := sendRpcRequestCallHelper[any](w, "authenticate", data, opts)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "controller:restart", wshserver.BlockRestartCommand
|
// command "controllerinput", wshserver.ControllerInputCommand
|
||||||
func BlockRestartCommand(w *wshutil.WshRpc, data wshrpc.CommandBlockRestartData, opts *wshrpc.WshRpcCommandOpts) error {
|
func ControllerInputCommand(w *wshutil.WshRpc, data wshrpc.CommandBlockInputData, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
_, err := sendRpcRequestCallHelper[any](w, "controller:restart", data, opts)
|
_, err := sendRpcRequestCallHelper[any](w, "controllerinput", data, opts)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "controllerrestart", wshserver.ControllerRestartCommand
|
||||||
|
func ControllerRestartCommand(w *wshutil.WshRpc, data wshrpc.CommandBlockRestartData, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
|
_, err := sendRpcRequestCallHelper[any](w, "controllerrestart", data, opts)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "createblock", wshserver.CreateBlockCommand
|
// command "createblock", wshserver.CreateBlockCommand
|
||||||
func CreateBlockCommand(w *wshutil.WshRpc, data wshrpc.CommandCreateBlockData, opts *wshrpc.WshRpcCommandOpts) (*waveobj.ORef, error) {
|
func CreateBlockCommand(w *wshutil.WshRpc, data wshrpc.CommandCreateBlockData, opts *wshrpc.WshRpcCommandOpts) (waveobj.ORef, error) {
|
||||||
resp, err := sendRpcRequestCallHelper[*waveobj.ORef](w, "createblock", data, opts)
|
resp, err := sendRpcRequestCallHelper[waveobj.ORef](w, "createblock", data, opts)
|
||||||
return resp, err
|
return resp, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -36,27 +41,57 @@ func DeleteBlockCommand(w *wshutil.WshRpc, data wshrpc.CommandDeleteBlockData, o
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "file:append", wshserver.AppendFileCommand
|
// command "eventpublish", wshserver.EventPublishCommand
|
||||||
func AppendFileCommand(w *wshutil.WshRpc, data wshrpc.CommandFileData, opts *wshrpc.WshRpcCommandOpts) error {
|
func EventPublishCommand(w *wshutil.WshRpc, data wshrpc.WaveEvent, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
_, err := sendRpcRequestCallHelper[any](w, "file:append", data, opts)
|
_, err := sendRpcRequestCallHelper[any](w, "eventpublish", data, opts)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "file:appendijson", wshserver.AppendIJsonCommand
|
// command "eventrecv", wshserver.EventRecvCommand
|
||||||
func AppendIJsonCommand(w *wshutil.WshRpc, data wshrpc.CommandAppendIJsonData, opts *wshrpc.WshRpcCommandOpts) error {
|
func EventRecvCommand(w *wshutil.WshRpc, data wshrpc.WaveEvent, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
_, err := sendRpcRequestCallHelper[any](w, "file:appendijson", data, opts)
|
_, err := sendRpcRequestCallHelper[any](w, "eventrecv", data, opts)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "file:read", wshserver.ReadFile
|
// command "eventsub", wshserver.EventSubCommand
|
||||||
func ReadFile(w *wshutil.WshRpc, data wshrpc.CommandFileData, opts *wshrpc.WshRpcCommandOpts) (string, error) {
|
func EventSubCommand(w *wshutil.WshRpc, data wshrpc.SubscriptionRequest, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
resp, err := sendRpcRequestCallHelper[string](w, "file:read", data, opts)
|
_, err := sendRpcRequestCallHelper[any](w, "eventsub", data, opts)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "eventunsub", wshserver.EventUnsubCommand
|
||||||
|
func EventUnsubCommand(w *wshutil.WshRpc, data wshrpc.SubscriptionRequest, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
|
_, err := sendRpcRequestCallHelper[any](w, "eventunsub", data, opts)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "eventunsuball", wshserver.EventUnsubAllCommand
|
||||||
|
func EventUnsubAllCommand(w *wshutil.WshRpc, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
|
_, err := sendRpcRequestCallHelper[any](w, "eventunsuball", nil, opts)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "fileappend", wshserver.FileAppendCommand
|
||||||
|
func FileAppendCommand(w *wshutil.WshRpc, data wshrpc.CommandFileData, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
|
_, err := sendRpcRequestCallHelper[any](w, "fileappend", data, opts)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "fileappendijson", wshserver.FileAppendIJsonCommand
|
||||||
|
func FileAppendIJsonCommand(w *wshutil.WshRpc, data wshrpc.CommandAppendIJsonData, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
|
_, err := sendRpcRequestCallHelper[any](w, "fileappendijson", data, opts)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// command "fileread", wshserver.FileReadCommand
|
||||||
|
func FileReadCommand(w *wshutil.WshRpc, data wshrpc.CommandFileData, opts *wshrpc.WshRpcCommandOpts) (string, error) {
|
||||||
|
resp, err := sendRpcRequestCallHelper[string](w, "fileread", data, opts)
|
||||||
return resp, err
|
return resp, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "file:write", wshserver.WriteFile
|
// command "filewrite", wshserver.FileWriteCommand
|
||||||
func WriteFile(w *wshutil.WshRpc, data wshrpc.CommandFileData, opts *wshrpc.WshRpcCommandOpts) error {
|
func FileWriteCommand(w *wshutil.WshRpc, data wshrpc.CommandFileData, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
_, err := sendRpcRequestCallHelper[any](w, "file:write", data, opts)
|
_, err := sendRpcRequestCallHelper[any](w, "filewrite", data, opts)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -84,20 +119,20 @@ func SetMetaCommand(w *wshutil.WshRpc, data wshrpc.CommandSetMetaData, opts *wsh
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "setview", wshserver.BlockSetViewCommand
|
// command "setview", wshserver.SetViewCommand
|
||||||
func BlockSetViewCommand(w *wshutil.WshRpc, data wshrpc.CommandBlockSetViewData, opts *wshrpc.WshRpcCommandOpts) error {
|
func SetViewCommand(w *wshutil.WshRpc, data wshrpc.CommandBlockSetViewData, opts *wshrpc.WshRpcCommandOpts) error {
|
||||||
_, err := sendRpcRequestCallHelper[any](w, "setview", data, opts)
|
_, err := sendRpcRequestCallHelper[any](w, "setview", data, opts)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// command "stream:waveai", wshserver.RespStreamWaveAi
|
// command "streamtest", wshserver.StreamTestCommand
|
||||||
func RespStreamWaveAi(w *wshutil.WshRpc, data waveai.OpenAiStreamRequest, opts *wshrpc.WshRpcCommandOpts) chan wshrpc.RespOrErrorUnion[waveai.OpenAIPacketType] {
|
func StreamTestCommand(w *wshutil.WshRpc, opts *wshrpc.WshRpcCommandOpts) chan wshrpc.RespOrErrorUnion[int] {
|
||||||
return sendRpcRequestResponseStreamHelper[waveai.OpenAIPacketType](w, "stream:waveai", data, opts)
|
|
||||||
}
|
|
||||||
|
|
||||||
// command "streamtest", wshserver.RespStreamTest
|
|
||||||
func RespStreamTest(w *wshutil.WshRpc, opts *wshrpc.WshRpcCommandOpts) chan wshrpc.RespOrErrorUnion[int] {
|
|
||||||
return sendRpcRequestResponseStreamHelper[int](w, "streamtest", nil, opts)
|
return sendRpcRequestResponseStreamHelper[int](w, "streamtest", nil, opts)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// command "streamwaveai", wshserver.StreamWaveAiCommand
|
||||||
|
func StreamWaveAiCommand(w *wshutil.WshRpc, data wshrpc.OpenAiStreamRequest, opts *wshrpc.WshRpcCommandOpts) chan wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType] {
|
||||||
|
return sendRpcRequestResponseStreamHelper[wshrpc.OpenAIPacketType](w, "streamwaveai", data, opts)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
116
pkg/wshrpc/wshrpcmeta.go
Normal file
116
pkg/wshrpc/wshrpcmeta.go
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
// Copyright 2024, Command Line Inc.
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
package wshrpc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type WshRpcMethodDecl struct {
|
||||||
|
Command string
|
||||||
|
CommandType string
|
||||||
|
MethodName string
|
||||||
|
CommandDataType reflect.Type
|
||||||
|
DefaultResponseDataType reflect.Type
|
||||||
|
}
|
||||||
|
|
||||||
|
var contextRType = reflect.TypeOf((*context.Context)(nil)).Elem()
|
||||||
|
var wshRpcInterfaceRType = reflect.TypeOf((*WshRpcInterface)(nil)).Elem()
|
||||||
|
|
||||||
|
func getWshCommandType(method reflect.Method) string {
|
||||||
|
if method.Type.NumOut() == 1 {
|
||||||
|
outType := method.Type.Out(0)
|
||||||
|
if outType.Kind() == reflect.Chan {
|
||||||
|
return RpcType_ResponseStream
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return RpcType_Call
|
||||||
|
}
|
||||||
|
|
||||||
|
func getWshMethodResponseType(commandType string, method reflect.Method) reflect.Type {
|
||||||
|
switch commandType {
|
||||||
|
case RpcType_ResponseStream:
|
||||||
|
if method.Type.NumOut() != 1 {
|
||||||
|
panic(fmt.Sprintf("method %q has invalid number of return values for response stream", method.Name))
|
||||||
|
}
|
||||||
|
outType := method.Type.Out(0)
|
||||||
|
if outType.Kind() != reflect.Chan {
|
||||||
|
panic(fmt.Sprintf("method %q has invalid return type %s for response stream", method.Name, outType))
|
||||||
|
}
|
||||||
|
elemType := outType.Elem()
|
||||||
|
if !strings.HasPrefix(elemType.Name(), "RespOrErrorUnion") {
|
||||||
|
panic(fmt.Sprintf("method %q has invalid return element type %s for response stream (should be RespOrErrorUnion)", method.Name, elemType))
|
||||||
|
}
|
||||||
|
respField, found := elemType.FieldByName("Response")
|
||||||
|
if !found {
|
||||||
|
panic(fmt.Sprintf("method %q has invalid return element type %s for response stream (missing Response field)", method.Name, elemType))
|
||||||
|
}
|
||||||
|
return respField.Type
|
||||||
|
case RpcType_Call:
|
||||||
|
if method.Type.NumOut() > 1 {
|
||||||
|
return method.Type.Out(0)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("unsupported command type %q", commandType))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func generateWshCommandDecl(method reflect.Method) *WshRpcMethodDecl {
|
||||||
|
if method.Type.NumIn() == 0 || method.Type.In(0) != contextRType {
|
||||||
|
panic(fmt.Sprintf("method %q does not have context as first argument", method.Name))
|
||||||
|
}
|
||||||
|
cmdStr := method.Name
|
||||||
|
decl := &WshRpcMethodDecl{}
|
||||||
|
// remove Command suffix
|
||||||
|
if !strings.HasSuffix(cmdStr, "Command") {
|
||||||
|
panic(fmt.Sprintf("method %q does not have Command suffix", cmdStr))
|
||||||
|
}
|
||||||
|
cmdStr = cmdStr[:len(cmdStr)-len("Command")]
|
||||||
|
decl.Command = strings.ToLower(cmdStr)
|
||||||
|
decl.CommandType = getWshCommandType(method)
|
||||||
|
decl.MethodName = method.Name
|
||||||
|
var cdataType reflect.Type
|
||||||
|
if method.Type.NumIn() > 1 {
|
||||||
|
cdataType = method.Type.In(1)
|
||||||
|
}
|
||||||
|
decl.CommandDataType = cdataType
|
||||||
|
decl.DefaultResponseDataType = getWshMethodResponseType(decl.CommandType, method)
|
||||||
|
return decl
|
||||||
|
}
|
||||||
|
|
||||||
|
func MakeMethodMapForImpl(impl any, declMap map[string]*WshRpcMethodDecl) map[string]reflect.Method {
|
||||||
|
rtype := reflect.TypeOf(impl)
|
||||||
|
rtnMap := make(map[string]reflect.Method)
|
||||||
|
for midx := 0; midx < rtype.NumMethod(); midx++ {
|
||||||
|
method := rtype.Method(midx)
|
||||||
|
if !strings.HasSuffix(method.Name, "Command") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
commandName := strings.ToLower(method.Name[:len(method.Name)-len("Command")])
|
||||||
|
decl := declMap[commandName]
|
||||||
|
if decl == nil {
|
||||||
|
log.Printf("WARNING: method %q does not match a command method", method.Name)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
rtnMap[commandName] = method
|
||||||
|
}
|
||||||
|
return rtnMap
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func GenerateWshCommandDeclMap() map[string]*WshRpcMethodDecl {
|
||||||
|
rtype := wshRpcInterfaceRType
|
||||||
|
rtnMap := make(map[string]*WshRpcMethodDecl)
|
||||||
|
for midx := 0; midx < rtype.NumMethod(); midx++ {
|
||||||
|
method := rtype.Method(midx)
|
||||||
|
decl := generateWshCommandDecl(method)
|
||||||
|
rtnMap[decl.Command] = decl
|
||||||
|
}
|
||||||
|
return rtnMap
|
||||||
|
}
|
@ -5,45 +5,77 @@
|
|||||||
package wshrpc
|
package wshrpc
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
|
||||||
"github.com/wavetermdev/thenextwave/pkg/ijson"
|
"github.com/wavetermdev/thenextwave/pkg/ijson"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/shellexec"
|
"github.com/wavetermdev/thenextwave/pkg/shellexec"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
RpcType_Call = "call" // single response (regular rpc)
|
||||||
|
RpcType_ResponseStream = "responsestream" // stream of responses (streaming rpc)
|
||||||
|
RpcType_StreamingRequest = "streamingrequest" // streaming request
|
||||||
|
RpcType_Complex = "complex" // streaming request/response
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
Command_Authenticate = "authenticate"
|
||||||
Command_Message = "message"
|
Command_Message = "message"
|
||||||
Command_SetView = "setview"
|
|
||||||
Command_SetMeta = "setmeta"
|
|
||||||
Command_GetMeta = "getmeta"
|
Command_GetMeta = "getmeta"
|
||||||
Command_BlockInput = "controller:input"
|
Command_SetMeta = "setmeta"
|
||||||
Command_Restart = "controller:restart"
|
Command_SetView = "setview"
|
||||||
Command_AppendFile = "file:append"
|
Command_ControllerInput = "controllerinput"
|
||||||
Command_AppendIJson = "file:appendijson"
|
Command_ControllerRestart = "controllerrestart"
|
||||||
|
Command_FileAppend = "fileappend"
|
||||||
|
Command_FileAppendIJson = "fileappendijson"
|
||||||
Command_ResolveIds = "resolveids"
|
Command_ResolveIds = "resolveids"
|
||||||
Command_CreateBlock = "createblock"
|
Command_CreateBlock = "createblock"
|
||||||
Command_DeleteBlock = "deleteblock"
|
Command_DeleteBlock = "deleteblock"
|
||||||
Command_WriteFile = "file:write"
|
Command_FileWrite = "filewrite"
|
||||||
Command_ReadFile = "file:read"
|
Command_FileRead = "fileread"
|
||||||
Command_StreamWaveAi = "stream:waveai"
|
Command_EventPublish = "eventpublish"
|
||||||
|
Command_EventRecv = "eventrecv"
|
||||||
|
Command_EventSub = "eventsub"
|
||||||
|
Command_EventUnsub = "eventunsub"
|
||||||
|
Command_EventUnsubAll = "eventunsuball"
|
||||||
|
Command_StreamTest = "streamtest"
|
||||||
|
Command_StreamWaveAi = "streamwaveai"
|
||||||
)
|
)
|
||||||
|
|
||||||
type MetaDataType = map[string]any
|
type MetaDataType = map[string]any
|
||||||
|
|
||||||
var DataTypeMap = map[string]reflect.Type{
|
|
||||||
"meta": reflect.TypeOf(MetaDataType{}),
|
|
||||||
"resolveidsrtn": reflect.TypeOf(CommandResolveIdsRtnData{}),
|
|
||||||
"oref": reflect.TypeOf(waveobj.ORef{}),
|
|
||||||
}
|
|
||||||
|
|
||||||
type RespOrErrorUnion[T any] struct {
|
type RespOrErrorUnion[T any] struct {
|
||||||
Response T
|
Response T
|
||||||
Error error
|
Error error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type WshRpcInterface interface {
|
||||||
|
AuthenticateCommand(ctx context.Context, data string) error
|
||||||
|
MessageCommand(ctx context.Context, data CommandMessageData) error
|
||||||
|
GetMetaCommand(ctx context.Context, data CommandGetMetaData) (MetaDataType, error)
|
||||||
|
SetMetaCommand(ctx context.Context, data CommandSetMetaData) error
|
||||||
|
SetViewCommand(ctx context.Context, data CommandBlockSetViewData) error
|
||||||
|
ControllerInputCommand(ctx context.Context, data CommandBlockInputData) error
|
||||||
|
ControllerRestartCommand(ctx context.Context, data CommandBlockRestartData) error
|
||||||
|
FileAppendCommand(ctx context.Context, data CommandFileData) error
|
||||||
|
FileAppendIJsonCommand(ctx context.Context, data CommandAppendIJsonData) error
|
||||||
|
ResolveIdsCommand(ctx context.Context, data CommandResolveIdsData) (CommandResolveIdsRtnData, error)
|
||||||
|
CreateBlockCommand(ctx context.Context, data CommandCreateBlockData) (waveobj.ORef, error)
|
||||||
|
DeleteBlockCommand(ctx context.Context, data CommandDeleteBlockData) error
|
||||||
|
FileWriteCommand(ctx context.Context, data CommandFileData) error
|
||||||
|
FileReadCommand(ctx context.Context, data CommandFileData) (string, error)
|
||||||
|
EventPublishCommand(ctx context.Context, data WaveEvent) error
|
||||||
|
EventRecvCommand(ctx context.Context, data WaveEvent) error
|
||||||
|
EventSubCommand(ctx context.Context, data SubscriptionRequest) error
|
||||||
|
EventUnsubCommand(ctx context.Context, data SubscriptionRequest) error
|
||||||
|
EventUnsubAllCommand(ctx context.Context) error
|
||||||
|
StreamTestCommand(ctx context.Context) chan RespOrErrorUnion[int]
|
||||||
|
StreamWaveAiCommand(ctx context.Context, request OpenAiStreamRequest) chan RespOrErrorUnion[OpenAIPacketType]
|
||||||
|
}
|
||||||
|
|
||||||
// for frontend
|
// for frontend
|
||||||
type WshServerCommandMeta struct {
|
type WshServerCommandMeta struct {
|
||||||
CommandType string `json:"commandtype"`
|
CommandType string `json:"commandtype"`
|
||||||
@ -54,7 +86,13 @@ type WshRpcCommandOpts struct {
|
|||||||
NoResponse bool `json:"noresponse"`
|
NoResponse bool `json:"noresponse"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func HackRpcContextIntoData(dataPtr any, rpcContext wshutil.RpcContext) {
|
type RpcContext struct {
|
||||||
|
BlockId string `json:"blockid,omitempty"`
|
||||||
|
TabId string `json:"tabid,omitempty"`
|
||||||
|
WindowId string `json:"windowid,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func HackRpcContextIntoData(dataPtr any, rpcContext RpcContext) {
|
||||||
dataVal := reflect.ValueOf(dataPtr).Elem()
|
dataVal := reflect.ValueOf(dataPtr).Elem()
|
||||||
dataType := dataVal.Type()
|
dataType := dataVal.Type()
|
||||||
for i := 0; i < dataVal.NumField(); i++ {
|
for i := 0; i < dataVal.NumField(); i++ {
|
||||||
@ -141,3 +179,54 @@ type CommandAppendIJsonData struct {
|
|||||||
type CommandDeleteBlockData struct {
|
type CommandDeleteBlockData struct {
|
||||||
BlockId string `json:"blockid" wshcontext:"BlockId"`
|
BlockId string `json:"blockid" wshcontext:"BlockId"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type WaveEvent struct {
|
||||||
|
Event string `json:"event"`
|
||||||
|
Scopes []string `json:"scopes,omitempty"`
|
||||||
|
Sender string `json:"sender,omitempty"`
|
||||||
|
Data any `json:"data,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SubscriptionRequest struct {
|
||||||
|
Event string `json:"event"`
|
||||||
|
Scopes []string `json:"scopes,omitempty"`
|
||||||
|
AllScopes bool `json:"allscopes,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type OpenAiStreamRequest struct {
|
||||||
|
ClientId string `json:"clientid,omitempty"`
|
||||||
|
Opts *OpenAIOptsType `json:"opts"`
|
||||||
|
Prompt []OpenAIPromptMessageType `json:"prompt"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type OpenAIPromptMessageType struct {
|
||||||
|
Role string `json:"role"`
|
||||||
|
Content string `json:"content"`
|
||||||
|
Name string `json:"name,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type OpenAIOptsType struct {
|
||||||
|
Model string `json:"model"`
|
||||||
|
APIToken string `json:"apitoken"`
|
||||||
|
BaseURL string `json:"baseurl,omitempty"`
|
||||||
|
MaxTokens int `json:"maxtokens,omitempty"`
|
||||||
|
MaxChoices int `json:"maxchoices,omitempty"`
|
||||||
|
Timeout int `json:"timeout,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type OpenAIPacketType struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Model string `json:"model,omitempty"`
|
||||||
|
Created int64 `json:"created,omitempty"`
|
||||||
|
FinishReason string `json:"finish_reason,omitempty"`
|
||||||
|
Usage *OpenAIUsageType `json:"usage,omitempty"`
|
||||||
|
Index int `json:"index,omitempty"`
|
||||||
|
Text string `json:"text,omitempty"`
|
||||||
|
Error string `json:"error,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type OpenAIUsageType struct {
|
||||||
|
PromptTokens int `json:"prompt_tokens,omitempty"`
|
||||||
|
CompletionTokens int `json:"completion_tokens,omitempty"`
|
||||||
|
TotalTokens int `json:"total_tokens,omitempty"`
|
||||||
|
}
|
||||||
|
@ -11,7 +11,6 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"io/fs"
|
"io/fs"
|
||||||
"log"
|
"log"
|
||||||
"reflect"
|
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -20,45 +19,26 @@ import (
|
|||||||
"github.com/wavetermdev/thenextwave/pkg/filestore"
|
"github.com/wavetermdev/thenextwave/pkg/filestore"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/waveai"
|
"github.com/wavetermdev/thenextwave/pkg/waveai"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
"github.com/wavetermdev/thenextwave/pkg/waveobj"
|
||||||
|
"github.com/wavetermdev/thenextwave/pkg/wps"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
"github.com/wavetermdev/thenextwave/pkg/wshutil"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
"github.com/wavetermdev/thenextwave/pkg/wstore"
|
||||||
)
|
)
|
||||||
|
|
||||||
var RespStreamTest_MethodDecl = &WshServerMethodDecl{
|
func (ws *WshServer) AuthenticateCommand(ctx context.Context, data string) error {
|
||||||
Command: "streamtest",
|
w := wshutil.GetWshRpcFromContext(ctx)
|
||||||
CommandType: wshutil.RpcType_ResponseStream,
|
if w == nil {
|
||||||
MethodName: "RespStreamTest",
|
return fmt.Errorf("no wshrpc in context")
|
||||||
Method: reflect.ValueOf(WshServerImpl.RespStreamTest),
|
|
||||||
CommandDataType: nil,
|
|
||||||
DefaultResponseDataType: reflect.TypeOf((int)(0)),
|
|
||||||
}
|
}
|
||||||
|
newCtx, err := wshutil.ValidateAndExtractRpcContextFromToken(data)
|
||||||
var RespStreamWaveAi_MethodDecl = &WshServerMethodDecl{
|
if err != nil {
|
||||||
Command: wshrpc.Command_StreamWaveAi,
|
return fmt.Errorf("error validating token: %w", err)
|
||||||
CommandType: wshutil.RpcType_ResponseStream,
|
|
||||||
MethodName: "RespStreamWaveAi",
|
|
||||||
Method: reflect.ValueOf(WshServerImpl.RespStreamWaveAi),
|
|
||||||
CommandDataType: reflect.TypeOf(waveai.OpenAiStreamRequest{}),
|
|
||||||
DefaultResponseDataType: reflect.TypeOf(waveai.OpenAIPacketType{}),
|
|
||||||
}
|
}
|
||||||
|
if newCtx == nil {
|
||||||
var WshServerCommandToDeclMap = map[string]*WshServerMethodDecl{
|
return fmt.Errorf("no context found in jwt token")
|
||||||
wshrpc.Command_Message: GetWshServerMethod(wshrpc.Command_Message, wshutil.RpcType_Call, "MessageCommand", WshServerImpl.MessageCommand),
|
}
|
||||||
wshrpc.Command_SetView: GetWshServerMethod(wshrpc.Command_SetView, wshutil.RpcType_Call, "BlockSetViewCommand", WshServerImpl.BlockSetViewCommand),
|
w.SetRpcContext(*newCtx)
|
||||||
wshrpc.Command_SetMeta: GetWshServerMethod(wshrpc.Command_SetMeta, wshutil.RpcType_Call, "SetMetaCommand", WshServerImpl.SetMetaCommand),
|
return nil
|
||||||
wshrpc.Command_GetMeta: GetWshServerMethod(wshrpc.Command_GetMeta, wshutil.RpcType_Call, "GetMetaCommand", WshServerImpl.GetMetaCommand),
|
|
||||||
wshrpc.Command_ResolveIds: GetWshServerMethod(wshrpc.Command_ResolveIds, wshutil.RpcType_Call, "ResolveIdsCommand", WshServerImpl.ResolveIdsCommand),
|
|
||||||
wshrpc.Command_CreateBlock: GetWshServerMethod(wshrpc.Command_CreateBlock, wshutil.RpcType_Call, "CreateBlockCommand", WshServerImpl.CreateBlockCommand),
|
|
||||||
wshrpc.Command_Restart: GetWshServerMethod(wshrpc.Command_Restart, wshutil.RpcType_Call, "BlockRestartCommand", WshServerImpl.BlockRestartCommand),
|
|
||||||
wshrpc.Command_BlockInput: GetWshServerMethod(wshrpc.Command_BlockInput, wshutil.RpcType_Call, "BlockInputCommand", WshServerImpl.BlockInputCommand),
|
|
||||||
wshrpc.Command_AppendFile: GetWshServerMethod(wshrpc.Command_AppendFile, wshutil.RpcType_Call, "AppendFileCommand", WshServerImpl.AppendFileCommand),
|
|
||||||
wshrpc.Command_AppendIJson: GetWshServerMethod(wshrpc.Command_AppendIJson, wshutil.RpcType_Call, "AppendIJsonCommand", WshServerImpl.AppendIJsonCommand),
|
|
||||||
wshrpc.Command_DeleteBlock: GetWshServerMethod(wshrpc.Command_DeleteBlock, wshutil.RpcType_Call, "DeleteBlockCommand", WshServerImpl.DeleteBlockCommand),
|
|
||||||
wshrpc.Command_WriteFile: GetWshServerMethod(wshrpc.Command_WriteFile, wshutil.RpcType_Call, "WriteFile", WshServerImpl.WriteFile),
|
|
||||||
wshrpc.Command_ReadFile: GetWshServerMethod(wshrpc.Command_ReadFile, wshutil.RpcType_Call, "ReadFile", WshServerImpl.ReadFile),
|
|
||||||
wshrpc.Command_StreamWaveAi: RespStreamWaveAi_MethodDecl,
|
|
||||||
"streamtest": RespStreamTest_MethodDecl,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// for testing
|
// for testing
|
||||||
@ -68,7 +48,7 @@ func (ws *WshServer) MessageCommand(ctx context.Context, data wshrpc.CommandMess
|
|||||||
}
|
}
|
||||||
|
|
||||||
// for testing
|
// for testing
|
||||||
func (ws *WshServer) RespStreamTest(ctx context.Context) chan wshrpc.RespOrErrorUnion[int] {
|
func (ws *WshServer) StreamTestCommand(ctx context.Context) chan wshrpc.RespOrErrorUnion[int] {
|
||||||
rtn := make(chan wshrpc.RespOrErrorUnion[int])
|
rtn := make(chan wshrpc.RespOrErrorUnion[int])
|
||||||
go func() {
|
go func() {
|
||||||
for i := 1; i <= 5; i++ {
|
for i := 1; i <= 5; i++ {
|
||||||
@ -80,7 +60,7 @@ func (ws *WshServer) RespStreamTest(ctx context.Context) chan wshrpc.RespOrError
|
|||||||
return rtn
|
return rtn
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WshServer) RespStreamWaveAi(ctx context.Context, request waveai.OpenAiStreamRequest) chan wshrpc.RespOrErrorUnion[waveai.OpenAIPacketType] {
|
func (ws *WshServer) StreamWaveAiCommand(ctx context.Context, request wshrpc.OpenAiStreamRequest) chan wshrpc.RespOrErrorUnion[wshrpc.OpenAIPacketType] {
|
||||||
if request.Opts.BaseURL == "" && request.Opts.APIToken == "" {
|
if request.Opts.BaseURL == "" && request.Opts.APIToken == "" {
|
||||||
return waveai.RunCloudCompletionStream(ctx, request)
|
return waveai.RunCloudCompletionStream(ctx, request)
|
||||||
}
|
}
|
||||||
@ -224,7 +204,7 @@ func (ws *WshServer) CreateBlockCommand(ctx context.Context, data wshrpc.Command
|
|||||||
return &waveobj.ORef{OType: wstore.OType_Block, OID: blockData.OID}, nil
|
return &waveobj.ORef{OType: wstore.OType_Block, OID: blockData.OID}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WshServer) BlockSetViewCommand(ctx context.Context, data wshrpc.CommandBlockSetViewData) error {
|
func (ws *WshServer) SetViewCommand(ctx context.Context, data wshrpc.CommandBlockSetViewData) error {
|
||||||
log.Printf("SETVIEW: %s | %q\n", data.BlockId, data.View)
|
log.Printf("SETVIEW: %s | %q\n", data.BlockId, data.View)
|
||||||
ctx = wstore.ContextWithUpdates(ctx)
|
ctx = wstore.ContextWithUpdates(ctx)
|
||||||
block, err := wstore.DBGet[*wstore.Block](ctx, data.BlockId)
|
block, err := wstore.DBGet[*wstore.Block](ctx, data.BlockId)
|
||||||
@ -241,7 +221,7 @@ func (ws *WshServer) BlockSetViewCommand(ctx context.Context, data wshrpc.Comman
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WshServer) BlockRestartCommand(ctx context.Context, data wshrpc.CommandBlockRestartData) error {
|
func (ws *WshServer) ControllerRestartCommand(ctx context.Context, data wshrpc.CommandBlockRestartData) error {
|
||||||
bc := blockcontroller.GetBlockController(data.BlockId)
|
bc := blockcontroller.GetBlockController(data.BlockId)
|
||||||
if bc == nil {
|
if bc == nil {
|
||||||
return fmt.Errorf("block controller not found for block %q", data.BlockId)
|
return fmt.Errorf("block controller not found for block %q", data.BlockId)
|
||||||
@ -249,7 +229,7 @@ func (ws *WshServer) BlockRestartCommand(ctx context.Context, data wshrpc.Comman
|
|||||||
return bc.RestartController()
|
return bc.RestartController()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WshServer) BlockInputCommand(ctx context.Context, data wshrpc.CommandBlockInputData) error {
|
func (ws *WshServer) ControllerInputCommand(ctx context.Context, data wshrpc.CommandBlockInputData) error {
|
||||||
bc := blockcontroller.GetBlockController(data.BlockId)
|
bc := blockcontroller.GetBlockController(data.BlockId)
|
||||||
if bc == nil {
|
if bc == nil {
|
||||||
return fmt.Errorf("block controller not found for block %q", data.BlockId)
|
return fmt.Errorf("block controller not found for block %q", data.BlockId)
|
||||||
@ -269,7 +249,7 @@ func (ws *WshServer) BlockInputCommand(ctx context.Context, data wshrpc.CommandB
|
|||||||
return bc.SendInput(inputUnion)
|
return bc.SendInput(inputUnion)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WshServer) WriteFile(ctx context.Context, data wshrpc.CommandFileData) error {
|
func (ws *WshServer) FileWriteCommand(ctx context.Context, data wshrpc.CommandFileData) error {
|
||||||
dataBuf, err := base64.StdEncoding.DecodeString(data.Data64)
|
dataBuf, err := base64.StdEncoding.DecodeString(data.Data64)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("error decoding data64: %w", err)
|
return fmt.Errorf("error decoding data64: %w", err)
|
||||||
@ -290,7 +270,7 @@ func (ws *WshServer) WriteFile(ctx context.Context, data wshrpc.CommandFileData)
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WshServer) ReadFile(ctx context.Context, data wshrpc.CommandFileData) (string, error) {
|
func (ws *WshServer) FileReadCommand(ctx context.Context, data wshrpc.CommandFileData) (string, error) {
|
||||||
_, dataBuf, err := filestore.WFS.ReadFile(ctx, data.ZoneId, data.FileName)
|
_, dataBuf, err := filestore.WFS.ReadFile(ctx, data.ZoneId, data.FileName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", fmt.Errorf("error reading blockfile: %w", err)
|
return "", fmt.Errorf("error reading blockfile: %w", err)
|
||||||
@ -298,7 +278,7 @@ func (ws *WshServer) ReadFile(ctx context.Context, data wshrpc.CommandFileData)
|
|||||||
return base64.StdEncoding.EncodeToString(dataBuf), nil
|
return base64.StdEncoding.EncodeToString(dataBuf), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WshServer) AppendFileCommand(ctx context.Context, data wshrpc.CommandFileData) error {
|
func (ws *WshServer) FileAppendCommand(ctx context.Context, data wshrpc.CommandFileData) error {
|
||||||
dataBuf, err := base64.StdEncoding.DecodeString(data.Data64)
|
dataBuf, err := base64.StdEncoding.DecodeString(data.Data64)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("error decoding data64: %w", err)
|
return fmt.Errorf("error decoding data64: %w", err)
|
||||||
@ -320,7 +300,7 @@ func (ws *WshServer) AppendFileCommand(ctx context.Context, data wshrpc.CommandF
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ws *WshServer) AppendIJsonCommand(ctx context.Context, data wshrpc.CommandAppendIJsonData) error {
|
func (ws *WshServer) FileAppendIJsonCommand(ctx context.Context, data wshrpc.CommandAppendIJsonData) error {
|
||||||
tryCreate := true
|
tryCreate := true
|
||||||
if data.FileName == blockcontroller.BlockFile_Html && tryCreate {
|
if data.FileName == blockcontroller.BlockFile_Html && tryCreate {
|
||||||
err := filestore.WFS.MakeFile(ctx, data.ZoneId, data.FileName, nil, filestore.FileOptsType{MaxSize: blockcontroller.DefaultHtmlMaxFileSize, IJson: true})
|
err := filestore.WFS.MakeFile(ctx, data.ZoneId, data.FileName, nil, filestore.FileOptsType{MaxSize: blockcontroller.DefaultHtmlMaxFileSize, IJson: true})
|
||||||
@ -378,3 +358,46 @@ func (ws *WshServer) DeleteBlockCommand(ctx context.Context, data wshrpc.Command
|
|||||||
sendWStoreUpdatesToEventBus(updates)
|
sendWStoreUpdatesToEventBus(updates)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (ws *WshServer) EventRecvCommand(ctx context.Context, data wshrpc.WaveEvent) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ws *WshServer) EventPublishCommand(ctx context.Context, data wshrpc.WaveEvent) error {
|
||||||
|
wrpc := wshutil.GetWshRpcFromContext(ctx)
|
||||||
|
if wrpc == nil {
|
||||||
|
return fmt.Errorf("no wshrpc in context")
|
||||||
|
}
|
||||||
|
if data.Sender == "" {
|
||||||
|
data.Sender = wrpc.ClientId()
|
||||||
|
}
|
||||||
|
wps.Broker.Publish(data)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ws *WshServer) EventSubCommand(ctx context.Context, data wshrpc.SubscriptionRequest) error {
|
||||||
|
wrpc := wshutil.GetWshRpcFromContext(ctx)
|
||||||
|
if wrpc == nil {
|
||||||
|
return fmt.Errorf("no wshrpc in context")
|
||||||
|
}
|
||||||
|
wps.Broker.Subscribe(wrpc, data)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ws *WshServer) EventUnsubCommand(ctx context.Context, data wshrpc.SubscriptionRequest) error {
|
||||||
|
wrpc := wshutil.GetWshRpcFromContext(ctx)
|
||||||
|
if wrpc == nil {
|
||||||
|
return fmt.Errorf("no wshrpc in context")
|
||||||
|
}
|
||||||
|
wps.Broker.Unsubscribe(wrpc, data)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ws *WshServer) EventUnsubAllCommand(ctx context.Context) error {
|
||||||
|
wrpc := wshutil.GetWshRpcFromContext(ctx)
|
||||||
|
if wrpc == nil {
|
||||||
|
return fmt.Errorf("no wshrpc in context")
|
||||||
|
}
|
||||||
|
wps.Broker.UnsubscribeAll(wrpc)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
@ -10,9 +10,7 @@ import (
|
|||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
"reflect"
|
"reflect"
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/golang-jwt/jwt/v5"
|
|
||||||
"github.com/wavetermdev/thenextwave/pkg/util/utilfn"
|
"github.com/wavetermdev/thenextwave/pkg/util/utilfn"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wavebase"
|
"github.com/wavetermdev/thenextwave/pkg/wavebase"
|
||||||
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
@ -41,6 +39,7 @@ type WshServerMethodDecl struct {
|
|||||||
|
|
||||||
var WshServerImpl = WshServer{}
|
var WshServerImpl = WshServer{}
|
||||||
var contextRType = reflect.TypeOf((*context.Context)(nil)).Elem()
|
var contextRType = reflect.TypeOf((*context.Context)(nil)).Elem()
|
||||||
|
var wshCommandDeclMap = wshrpc.GenerateWshCommandDeclMap()
|
||||||
|
|
||||||
func GetWshServerMethod(command string, commandType string, methodName string, methodFunc any) *WshServerMethodDecl {
|
func GetWshServerMethod(command string, commandType string, methodName string, methodFunc any) *WshServerMethodDecl {
|
||||||
methodVal := reflect.ValueOf(methodFunc)
|
methodVal := reflect.ValueOf(methodFunc)
|
||||||
@ -55,12 +54,16 @@ func GetWshServerMethod(command string, commandType string, methodName string, m
|
|||||||
if methodType.NumOut() > 1 {
|
if methodType.NumOut() > 1 {
|
||||||
defResponseType = methodType.Out(0)
|
defResponseType = methodType.Out(0)
|
||||||
}
|
}
|
||||||
|
var cdataType reflect.Type
|
||||||
|
if methodType.NumIn() > 1 {
|
||||||
|
cdataType = methodType.In(1)
|
||||||
|
}
|
||||||
rtn := &WshServerMethodDecl{
|
rtn := &WshServerMethodDecl{
|
||||||
Command: command,
|
Command: command,
|
||||||
CommandType: commandType,
|
CommandType: commandType,
|
||||||
MethodName: methodName,
|
MethodName: methodName,
|
||||||
Method: methodVal,
|
Method: methodVal,
|
||||||
CommandDataType: methodType.In(1),
|
CommandDataType: cdataType,
|
||||||
DefaultResponseDataType: defResponseType,
|
DefaultResponseDataType: defResponseType,
|
||||||
}
|
}
|
||||||
return rtn
|
return rtn
|
||||||
@ -89,7 +92,7 @@ func decodeRtnVals(rtnVals []reflect.Value) (any, error) {
|
|||||||
|
|
||||||
func mainWshServerHandler(handler *wshutil.RpcResponseHandler) bool {
|
func mainWshServerHandler(handler *wshutil.RpcResponseHandler) bool {
|
||||||
command := handler.GetCommand()
|
command := handler.GetCommand()
|
||||||
methodDecl := WshServerCommandToDeclMap[command]
|
methodDecl := wshCommandDeclMap[command]
|
||||||
if methodDecl == nil {
|
if methodDecl == nil {
|
||||||
handler.SendResponseError(fmt.Errorf("command %q not found", command))
|
handler.SendResponseError(fmt.Errorf("command %q not found", command))
|
||||||
return true
|
return true
|
||||||
@ -106,8 +109,18 @@ func mainWshServerHandler(handler *wshutil.RpcResponseHandler) bool {
|
|||||||
wshrpc.HackRpcContextIntoData(commandData, handler.GetRpcContext())
|
wshrpc.HackRpcContextIntoData(commandData, handler.GetRpcContext())
|
||||||
callParams = append(callParams, reflect.ValueOf(commandData).Elem())
|
callParams = append(callParams, reflect.ValueOf(commandData).Elem())
|
||||||
}
|
}
|
||||||
if methodDecl.CommandType == wshutil.RpcType_Call {
|
implVal := reflect.ValueOf(&WshServerImpl)
|
||||||
rtnVals := methodDecl.Method.Call(callParams)
|
implMethod := implVal.MethodByName(methodDecl.MethodName)
|
||||||
|
if !implMethod.IsValid() {
|
||||||
|
if !handler.NeedsResponse() {
|
||||||
|
// we also send an out of band message here since this is likely unexpected and will require debugging
|
||||||
|
handler.SendMessage(fmt.Sprintf("command %q method %q not found", handler.GetCommand(), methodDecl.MethodName))
|
||||||
|
}
|
||||||
|
handler.SendResponseError(fmt.Errorf("method %q not found", methodDecl.MethodName))
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if methodDecl.CommandType == wshrpc.RpcType_Call {
|
||||||
|
rtnVals := implMethod.Call(callParams)
|
||||||
rtnData, rtnErr := decodeRtnVals(rtnVals)
|
rtnData, rtnErr := decodeRtnVals(rtnVals)
|
||||||
if rtnErr != nil {
|
if rtnErr != nil {
|
||||||
handler.SendResponseError(rtnErr)
|
handler.SendResponseError(rtnErr)
|
||||||
@ -115,8 +128,8 @@ func mainWshServerHandler(handler *wshutil.RpcResponseHandler) bool {
|
|||||||
}
|
}
|
||||||
handler.SendResponse(rtnData, true)
|
handler.SendResponse(rtnData, true)
|
||||||
return true
|
return true
|
||||||
} else if methodDecl.CommandType == wshutil.RpcType_ResponseStream {
|
} else if methodDecl.CommandType == wshrpc.RpcType_ResponseStream {
|
||||||
rtnVals := methodDecl.Method.Call(callParams)
|
rtnVals := implMethod.Call(callParams)
|
||||||
rtnChVal := rtnVals[0]
|
rtnChVal := rtnVals[0]
|
||||||
if rtnChVal.IsNil() {
|
if rtnChVal.IsNil() {
|
||||||
handler.SendResponse(nil, true)
|
handler.SendResponse(nil, true)
|
||||||
@ -163,7 +176,7 @@ func runWshRpcWithStream(conn net.Conn) {
|
|||||||
outputCh := make(chan []byte, DefaultOutputChSize)
|
outputCh := make(chan []byte, DefaultOutputChSize)
|
||||||
go wshutil.AdaptMsgChToStream(outputCh, conn)
|
go wshutil.AdaptMsgChToStream(outputCh, conn)
|
||||||
go wshutil.AdaptStreamToMsgCh(conn, inputCh)
|
go wshutil.AdaptStreamToMsgCh(conn, inputCh)
|
||||||
wshutil.MakeWshRpc(inputCh, outputCh, wshutil.RpcContext{}, mainWshServerHandler)
|
wshutil.MakeWshRpc(inputCh, outputCh, wshrpc.RpcContext{}, mainWshServerHandler)
|
||||||
}
|
}
|
||||||
|
|
||||||
func RunWshRpcOverListener(listener net.Listener) {
|
func RunWshRpcOverListener(listener net.Listener) {
|
||||||
@ -179,82 +192,6 @@ func RunWshRpcOverListener(listener net.Listener) {
|
|||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
func MakeClientJWTToken(rpcCtx wshutil.RpcContext, sockName string) (string, error) {
|
|
||||||
claims := jwt.MapClaims{}
|
|
||||||
claims["iat"] = time.Now().Unix()
|
|
||||||
claims["iss"] = "waveterm"
|
|
||||||
claims["sock"] = sockName
|
|
||||||
claims["exp"] = time.Now().Add(time.Hour * 24 * 365).Unix()
|
|
||||||
if rpcCtx.BlockId != "" {
|
|
||||||
claims["blockid"] = rpcCtx.BlockId
|
|
||||||
}
|
|
||||||
if rpcCtx.TabId != "" {
|
|
||||||
claims["tabid"] = rpcCtx.TabId
|
|
||||||
}
|
|
||||||
if rpcCtx.WindowId != "" {
|
|
||||||
claims["windowid"] = rpcCtx.WindowId
|
|
||||||
}
|
|
||||||
token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)
|
|
||||||
tokenStr, err := token.SignedString([]byte(wavebase.JwtSecret))
|
|
||||||
if err != nil {
|
|
||||||
return "", fmt.Errorf("error signing token: %w", err)
|
|
||||||
}
|
|
||||||
return tokenStr, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func ValidateAndExtractRpcContextFromToken(tokenStr string) (wshutil.RpcContext, error) {
|
|
||||||
parser := jwt.NewParser(jwt.WithValidMethods([]string{jwt.SigningMethodHS256.Name}))
|
|
||||||
token, err := parser.Parse(tokenStr, func(token *jwt.Token) (interface{}, error) {
|
|
||||||
return []byte(wavebase.JwtSecret), nil
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return wshutil.RpcContext{}, fmt.Errorf("error parsing token: %w", err)
|
|
||||||
}
|
|
||||||
claims, ok := token.Claims.(jwt.MapClaims)
|
|
||||||
if !ok {
|
|
||||||
return wshutil.RpcContext{}, fmt.Errorf("error getting claims from token")
|
|
||||||
}
|
|
||||||
// validate "exp" claim
|
|
||||||
if exp, ok := claims["exp"].(float64); ok {
|
|
||||||
if int64(exp) < time.Now().Unix() {
|
|
||||||
return wshutil.RpcContext{}, fmt.Errorf("token has expired")
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return wshutil.RpcContext{}, fmt.Errorf("exp claim is missing or invalid")
|
|
||||||
}
|
|
||||||
// validate "iss" claim
|
|
||||||
if iss, ok := claims["iss"].(string); ok {
|
|
||||||
if iss != "waveterm" {
|
|
||||||
return wshutil.RpcContext{}, fmt.Errorf("unexpected issuer: %s", iss)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return wshutil.RpcContext{}, fmt.Errorf("iss claim is missing or invalid")
|
|
||||||
}
|
|
||||||
rpcCtx := wshutil.RpcContext{}
|
|
||||||
rpcCtx.BlockId = claims["blockid"].(string)
|
|
||||||
rpcCtx.TabId = claims["tabid"].(string)
|
|
||||||
rpcCtx.WindowId = claims["windowid"].(string)
|
|
||||||
return rpcCtx, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func ExtractUnverifiedSocketName(tokenStr string) (string, error) {
|
|
||||||
// this happens on the client who does not have access to the secret key
|
|
||||||
// we want to read the claims without validating the signature
|
|
||||||
token, _, err := new(jwt.Parser).ParseUnverified(tokenStr, jwt.MapClaims{})
|
|
||||||
if err != nil {
|
|
||||||
return "", fmt.Errorf("error parsing token: %w", err)
|
|
||||||
}
|
|
||||||
claims, ok := token.Claims.(jwt.MapClaims)
|
|
||||||
if !ok {
|
|
||||||
return "", fmt.Errorf("error getting claims from token")
|
|
||||||
}
|
|
||||||
sockName, ok := claims["sock"].(string)
|
|
||||||
if !ok {
|
|
||||||
return "", fmt.Errorf("sock claim is missing or invalid")
|
|
||||||
}
|
|
||||||
return sockName, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func RunDomainSocketWshServer() error {
|
func RunDomainSocketWshServer() error {
|
||||||
sockName := wavebase.GetDomainSocketName()
|
sockName := wavebase.GetDomainSocketName()
|
||||||
listener, err := MakeUnixListener(sockName)
|
listener, err := MakeUnixListener(sockName)
|
||||||
@ -266,6 +203,6 @@ func RunDomainSocketWshServer() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func MakeWshServer(inputCh chan []byte, outputCh chan []byte, initialCtx wshutil.RpcContext) {
|
func MakeWshServer(inputCh chan []byte, outputCh chan []byte, initialCtx wshrpc.RpcContext) {
|
||||||
wshutil.MakeWshRpc(inputCh, outputCh, initialCtx, mainWshServerHandler)
|
wshutil.MakeWshRpc(inputCh, outputCh, initialCtx, mainWshServerHandler)
|
||||||
}
|
}
|
||||||
|
@ -15,19 +15,13 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
)
|
)
|
||||||
|
|
||||||
const DefaultTimeoutMs = 5000
|
const DefaultTimeoutMs = 5000
|
||||||
const RespChSize = 32
|
const RespChSize = 32
|
||||||
const DefaultMessageChSize = 32
|
const DefaultMessageChSize = 32
|
||||||
|
|
||||||
const (
|
|
||||||
RpcType_Call = "call" // single response (regular rpc)
|
|
||||||
RpcType_ResponseStream = "responsestream" // stream of responses (streaming rpc)
|
|
||||||
RpcType_StreamingRequest = "streamingrequest" // streaming request
|
|
||||||
RpcType_Complex = "complex" // streaming request/response
|
|
||||||
)
|
|
||||||
|
|
||||||
type ResponseFnType = func(any) error
|
type ResponseFnType = func(any) error
|
||||||
|
|
||||||
// returns true if handler is complete, false for an async handler
|
// returns true if handler is complete, false for an async handler
|
||||||
@ -115,17 +109,12 @@ func (r *RpcMessage) Validate() error {
|
|||||||
return fmt.Errorf("invalid packet: must have command, reqid, or resid set")
|
return fmt.Errorf("invalid packet: must have command, reqid, or resid set")
|
||||||
}
|
}
|
||||||
|
|
||||||
type RpcContext struct {
|
|
||||||
BlockId string `json:"blockid,omitempty"`
|
|
||||||
TabId string `json:"tabid,omitempty"`
|
|
||||||
WindowId string `json:"windowid,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type WshRpc struct {
|
type WshRpc struct {
|
||||||
Lock *sync.Mutex
|
Lock *sync.Mutex
|
||||||
|
clientId string
|
||||||
InputCh chan []byte
|
InputCh chan []byte
|
||||||
OutputCh chan []byte
|
OutputCh chan []byte
|
||||||
RpcContext *atomic.Pointer[RpcContext]
|
RpcContext *atomic.Pointer[wshrpc.RpcContext]
|
||||||
RpcMap map[string]*rpcData
|
RpcMap map[string]*rpcData
|
||||||
HandlerFn CommandHandlerFnType
|
HandlerFn CommandHandlerFnType
|
||||||
|
|
||||||
@ -139,13 +128,14 @@ type rpcData struct {
|
|||||||
|
|
||||||
// oscEsc is the OSC escape sequence to use for *sending* messages
|
// oscEsc is the OSC escape sequence to use for *sending* messages
|
||||||
// closes outputCh when inputCh is closed/done
|
// closes outputCh when inputCh is closed/done
|
||||||
func MakeWshRpc(inputCh chan []byte, outputCh chan []byte, rpcCtx RpcContext, commandHandlerFn CommandHandlerFnType) *WshRpc {
|
func MakeWshRpc(inputCh chan []byte, outputCh chan []byte, rpcCtx wshrpc.RpcContext, commandHandlerFn CommandHandlerFnType) *WshRpc {
|
||||||
rtn := &WshRpc{
|
rtn := &WshRpc{
|
||||||
Lock: &sync.Mutex{},
|
Lock: &sync.Mutex{},
|
||||||
|
clientId: uuid.New().String(),
|
||||||
InputCh: inputCh,
|
InputCh: inputCh,
|
||||||
OutputCh: outputCh,
|
OutputCh: outputCh,
|
||||||
RpcMap: make(map[string]*rpcData),
|
RpcMap: make(map[string]*rpcData),
|
||||||
RpcContext: &atomic.Pointer[RpcContext]{},
|
RpcContext: &atomic.Pointer[wshrpc.RpcContext]{},
|
||||||
HandlerFn: commandHandlerFn,
|
HandlerFn: commandHandlerFn,
|
||||||
ResponseHandlerMap: make(map[string]*RpcResponseHandler),
|
ResponseHandlerMap: make(map[string]*RpcResponseHandler),
|
||||||
}
|
}
|
||||||
@ -154,12 +144,30 @@ func MakeWshRpc(inputCh chan []byte, outputCh chan []byte, rpcCtx RpcContext, co
|
|||||||
return rtn
|
return rtn
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *WshRpc) GetRpcContext() RpcContext {
|
func (w *WshRpc) ClientId() string {
|
||||||
|
return w.clientId
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *WshRpc) SendEvent(event wshrpc.WaveEvent) {
|
||||||
|
// for wps compatibility
|
||||||
|
msg := &RpcMessage{
|
||||||
|
Command: wshrpc.Command_EventPublish,
|
||||||
|
Data: event,
|
||||||
|
}
|
||||||
|
barr, err := json.Marshal(msg)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("error marshalling event: %v\n", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.OutputCh <- barr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *WshRpc) GetRpcContext() wshrpc.RpcContext {
|
||||||
rtnPtr := w.RpcContext.Load()
|
rtnPtr := w.RpcContext.Load()
|
||||||
return *rtnPtr
|
return *rtnPtr
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *WshRpc) SetRpcContext(ctx RpcContext) {
|
func (w *WshRpc) SetRpcContext(ctx wshrpc.RpcContext) {
|
||||||
w.RpcContext.Store(&ctx)
|
w.RpcContext.Store(&ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -399,7 +407,7 @@ type RpcResponseHandler struct {
|
|||||||
reqId string
|
reqId string
|
||||||
command string
|
command string
|
||||||
commandData any
|
commandData any
|
||||||
rpcCtx RpcContext
|
rpcCtx wshrpc.RpcContext
|
||||||
canceled *atomic.Bool // canceled by requestor
|
canceled *atomic.Bool // canceled by requestor
|
||||||
done *atomic.Bool
|
done *atomic.Bool
|
||||||
}
|
}
|
||||||
@ -416,10 +424,25 @@ func (handler *RpcResponseHandler) GetCommandRawData() any {
|
|||||||
return handler.commandData
|
return handler.commandData
|
||||||
}
|
}
|
||||||
|
|
||||||
func (handler *RpcResponseHandler) GetRpcContext() RpcContext {
|
func (handler *RpcResponseHandler) GetRpcContext() wshrpc.RpcContext {
|
||||||
return handler.rpcCtx
|
return handler.rpcCtx
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (handler *RpcResponseHandler) NeedsResponse() bool {
|
||||||
|
return handler.reqId != ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (handler *RpcResponseHandler) SendMessage(msg string) {
|
||||||
|
rpcMsg := &RpcMessage{
|
||||||
|
Command: wshrpc.Command_Message,
|
||||||
|
Data: wshrpc.CommandMessageData{
|
||||||
|
Message: msg,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
msgBytes, _ := json.Marshal(rpcMsg) // will never fail
|
||||||
|
handler.w.OutputCh <- msgBytes
|
||||||
|
}
|
||||||
|
|
||||||
func (handler *RpcResponseHandler) SendResponse(data any, done bool) error {
|
func (handler *RpcResponseHandler) SendResponse(data any, done bool) error {
|
||||||
if handler.reqId == "" {
|
if handler.reqId == "" {
|
||||||
return nil // no response expected
|
return nil // no response expected
|
||||||
|
@ -14,7 +14,11 @@ import (
|
|||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/golang-jwt/jwt/v5"
|
||||||
|
"github.com/wavetermdev/thenextwave/pkg/wavebase"
|
||||||
|
"github.com/wavetermdev/thenextwave/pkg/wshrpc"
|
||||||
"golang.org/x/term"
|
"golang.org/x/term"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -180,7 +184,7 @@ func SetupTerminalRpcClient(handlerFn func(*RpcResponseHandler) bool) (*WshRpc,
|
|||||||
messageCh := make(chan []byte, 32)
|
messageCh := make(chan []byte, 32)
|
||||||
outputCh := make(chan []byte, 32)
|
outputCh := make(chan []byte, 32)
|
||||||
ptyBuf := MakePtyBuffer(WaveServerOSCPrefix, os.Stdin, messageCh)
|
ptyBuf := MakePtyBuffer(WaveServerOSCPrefix, os.Stdin, messageCh)
|
||||||
rpcClient := MakeWshRpc(messageCh, outputCh, RpcContext{}, handlerFn)
|
rpcClient := MakeWshRpc(messageCh, outputCh, wshrpc.RpcContext{}, handlerFn)
|
||||||
go func() {
|
go func() {
|
||||||
for msg := range outputCh {
|
for msg := range outputCh {
|
||||||
barr := EncodeWaveOSCBytes(WaveOSC, msg)
|
barr := EncodeWaveOSCBytes(WaveOSC, msg)
|
||||||
@ -189,3 +193,79 @@ func SetupTerminalRpcClient(handlerFn func(*RpcResponseHandler) bool) (*WshRpc,
|
|||||||
}()
|
}()
|
||||||
return rpcClient, ptyBuf
|
return rpcClient, ptyBuf
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func MakeClientJWTToken(rpcCtx wshrpc.RpcContext, sockName string) (string, error) {
|
||||||
|
claims := jwt.MapClaims{}
|
||||||
|
claims["iat"] = time.Now().Unix()
|
||||||
|
claims["iss"] = "waveterm"
|
||||||
|
claims["sock"] = sockName
|
||||||
|
claims["exp"] = time.Now().Add(time.Hour * 24 * 365).Unix()
|
||||||
|
if rpcCtx.BlockId != "" {
|
||||||
|
claims["blockid"] = rpcCtx.BlockId
|
||||||
|
}
|
||||||
|
if rpcCtx.TabId != "" {
|
||||||
|
claims["tabid"] = rpcCtx.TabId
|
||||||
|
}
|
||||||
|
if rpcCtx.WindowId != "" {
|
||||||
|
claims["windowid"] = rpcCtx.WindowId
|
||||||
|
}
|
||||||
|
token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)
|
||||||
|
tokenStr, err := token.SignedString([]byte(wavebase.JwtSecret))
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("error signing token: %w", err)
|
||||||
|
}
|
||||||
|
return tokenStr, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func ValidateAndExtractRpcContextFromToken(tokenStr string) (*wshrpc.RpcContext, error) {
|
||||||
|
parser := jwt.NewParser(jwt.WithValidMethods([]string{jwt.SigningMethodHS256.Name}))
|
||||||
|
token, err := parser.Parse(tokenStr, func(token *jwt.Token) (interface{}, error) {
|
||||||
|
return []byte(wavebase.JwtSecret), nil
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("error parsing token: %w", err)
|
||||||
|
}
|
||||||
|
claims, ok := token.Claims.(jwt.MapClaims)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("error getting claims from token")
|
||||||
|
}
|
||||||
|
// validate "exp" claim
|
||||||
|
if exp, ok := claims["exp"].(float64); ok {
|
||||||
|
if int64(exp) < time.Now().Unix() {
|
||||||
|
return nil, fmt.Errorf("token has expired")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return nil, fmt.Errorf("exp claim is missing or invalid")
|
||||||
|
}
|
||||||
|
// validate "iss" claim
|
||||||
|
if iss, ok := claims["iss"].(string); ok {
|
||||||
|
if iss != "waveterm" {
|
||||||
|
return nil, fmt.Errorf("unexpected issuer: %s", iss)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return nil, fmt.Errorf("iss claim is missing or invalid")
|
||||||
|
}
|
||||||
|
rpcCtx := &wshrpc.RpcContext{}
|
||||||
|
rpcCtx.BlockId = claims["blockid"].(string)
|
||||||
|
rpcCtx.TabId = claims["tabid"].(string)
|
||||||
|
rpcCtx.WindowId = claims["windowid"].(string)
|
||||||
|
return rpcCtx, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func ExtractUnverifiedSocketName(tokenStr string) (string, error) {
|
||||||
|
// this happens on the client who does not have access to the secret key
|
||||||
|
// we want to read the claims without validating the signature
|
||||||
|
token, _, err := new(jwt.Parser).ParseUnverified(tokenStr, jwt.MapClaims{})
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("error parsing token: %w", err)
|
||||||
|
}
|
||||||
|
claims, ok := token.Claims.(jwt.MapClaims)
|
||||||
|
if !ok {
|
||||||
|
return "", fmt.Errorf("error getting claims from token")
|
||||||
|
}
|
||||||
|
sockName, ok := claims["sock"].(string)
|
||||||
|
if !ok {
|
||||||
|
return "", fmt.Errorf("sock claim is missing or invalid")
|
||||||
|
}
|
||||||
|
return sockName, nil
|
||||||
|
}
|
||||||
|
39
yarn.lock
39
yarn.lock
@ -3546,6 +3546,31 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/color-convert@npm:*":
|
||||||
|
version: 2.0.3
|
||||||
|
resolution: "@types/color-convert@npm:2.0.3"
|
||||||
|
dependencies:
|
||||||
|
"@types/color-name": "npm:*"
|
||||||
|
checksum: 10c0/a5870547660f426cddd76b54e942703e29c3b43fc26b1ba567e10b9707d144b7d8863e0af7affd9c3391815c06582571f43835c71ede270a6c58949155d18b77
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/color-name@npm:*":
|
||||||
|
version: 1.1.4
|
||||||
|
resolution: "@types/color-name@npm:1.1.4"
|
||||||
|
checksum: 10c0/11a5b67408a53a972fa98e4bbe2b0ff4cb74a3b3abb5f250cb5ec7b055a45aa8e00ddaf39b8327ef683ede9b2ff9b3ee9d25cd708d12b1b6a9aee5e8e6002920
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/color@npm:^3.0.6":
|
||||||
|
version: 3.0.6
|
||||||
|
resolution: "@types/color@npm:3.0.6"
|
||||||
|
dependencies:
|
||||||
|
"@types/color-convert": "npm:*"
|
||||||
|
checksum: 10c0/79267eeb67f9d11761aecee36bb1503fb8daa699b9ae7e036fc23a74380e5b130c5c0f6d7adafabba89256e46f36ee4d3e28e0ac7e107e8258550eae7d091acf
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@types/connect@npm:*":
|
"@types/connect@npm:*":
|
||||||
version: 3.4.38
|
version: 3.4.38
|
||||||
resolution: "@types/connect@npm:3.4.38"
|
resolution: "@types/connect@npm:3.4.38"
|
||||||
@ -5258,7 +5283,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"color-string@npm:^1.6.0":
|
"color-string@npm:^1.6.0, color-string@npm:^1.9.0":
|
||||||
version: 1.9.1
|
version: 1.9.1
|
||||||
resolution: "color-string@npm:1.9.1"
|
resolution: "color-string@npm:1.9.1"
|
||||||
dependencies:
|
dependencies:
|
||||||
@ -5278,6 +5303,16 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"color@npm:^4.2.3":
|
||||||
|
version: 4.2.3
|
||||||
|
resolution: "color@npm:4.2.3"
|
||||||
|
dependencies:
|
||||||
|
color-convert: "npm:^2.0.1"
|
||||||
|
color-string: "npm:^1.9.0"
|
||||||
|
checksum: 10c0/7fbe7cfb811054c808349de19fb380252e5e34e61d7d168ec3353e9e9aacb1802674bddc657682e4e9730c2786592a4de6f8283e7e0d3870b829bb0b7b2f6118
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"colorspace@npm:1.1.x":
|
"colorspace@npm:1.1.x":
|
||||||
version: 1.1.4
|
version: 1.1.4
|
||||||
resolution: "colorspace@npm:1.1.4"
|
resolution: "colorspace@npm:1.1.4"
|
||||||
@ -11768,6 +11803,7 @@ __metadata:
|
|||||||
"@table-nav/core": "npm:^0.0.7"
|
"@table-nav/core": "npm:^0.0.7"
|
||||||
"@table-nav/react": "npm:^0.0.7"
|
"@table-nav/react": "npm:^0.0.7"
|
||||||
"@tanstack/react-table": "npm:^8.19.3"
|
"@tanstack/react-table": "npm:^8.19.3"
|
||||||
|
"@types/color": "npm:^3.0.6"
|
||||||
"@types/electron": "npm:^1.6.10"
|
"@types/electron": "npm:^1.6.10"
|
||||||
"@types/node": "npm:^20.14.12"
|
"@types/node": "npm:^20.14.12"
|
||||||
"@types/papaparse": "npm:^5"
|
"@types/papaparse": "npm:^5"
|
||||||
@ -11783,6 +11819,7 @@ __metadata:
|
|||||||
"@xterm/xterm": "npm:^5.5.0"
|
"@xterm/xterm": "npm:^5.5.0"
|
||||||
base64-js: "npm:^1.5.1"
|
base64-js: "npm:^1.5.1"
|
||||||
clsx: "npm:^2.1.1"
|
clsx: "npm:^2.1.1"
|
||||||
|
color: "npm:^4.2.3"
|
||||||
dayjs: "npm:^1.11.12"
|
dayjs: "npm:^1.11.12"
|
||||||
electron: "npm:^31.3.0"
|
electron: "npm:^31.3.0"
|
||||||
electron-builder: "npm:^24.13.3"
|
electron-builder: "npm:^24.13.3"
|
||||||
|
Loading…
Reference in New Issue
Block a user