2024-11-02 18:58:13 +01:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
package vdomclient
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/wavetermdev/waveterm/pkg/vdom"
|
|
|
|
"github.com/wavetermdev/waveterm/pkg/wshrpc"
|
|
|
|
)
|
|
|
|
|
|
|
|
type VDomServerImpl struct {
|
|
|
|
Client *Client
|
|
|
|
BlockId string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (*VDomServerImpl) WshServerImpl() {}
|
|
|
|
|
2024-11-06 00:52:59 +01:00
|
|
|
func (impl *VDomServerImpl) VDomRenderCommand(ctx context.Context, feUpdate vdom.VDomFrontendUpdate) chan wshrpc.RespOrErrorUnion[*vdom.VDomBackendUpdate] {
|
|
|
|
respChan := make(chan wshrpc.RespOrErrorUnion[*vdom.VDomBackendUpdate], 5)
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
|
|
|
log.Printf("panic in VDomRenderCommand: %v\n", r)
|
|
|
|
respChan <- wshrpc.RespOrErrorUnion[*vdom.VDomBackendUpdate]{
|
|
|
|
Error: fmt.Errorf("internal error: %v", r),
|
|
|
|
}
|
|
|
|
close(respChan)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2024-11-02 18:58:13 +01:00
|
|
|
if feUpdate.Dispose {
|
2024-11-06 00:52:59 +01:00
|
|
|
defer close(respChan)
|
2024-11-02 18:58:13 +01:00
|
|
|
log.Printf("got dispose from frontend\n")
|
|
|
|
impl.Client.doShutdown("got dispose from frontend")
|
2024-11-06 00:52:59 +01:00
|
|
|
return respChan
|
2024-11-02 18:58:13 +01:00
|
|
|
}
|
2024-11-06 00:52:59 +01:00
|
|
|
|
2024-11-02 18:58:13 +01:00
|
|
|
if impl.Client.GetIsDone() {
|
2024-11-06 00:52:59 +01:00
|
|
|
close(respChan)
|
|
|
|
return respChan
|
2024-11-02 18:58:13 +01:00
|
|
|
}
|
2024-11-06 00:52:59 +01:00
|
|
|
|
2024-11-06 08:07:45 +01:00
|
|
|
impl.Client.Root.RenderTs = feUpdate.Ts
|
|
|
|
|
2024-11-02 18:58:13 +01:00
|
|
|
// set atoms
|
|
|
|
for _, ss := range feUpdate.StateSync {
|
|
|
|
impl.Client.Root.SetAtomVal(ss.Atom, ss.Value, false)
|
|
|
|
}
|
|
|
|
// run events
|
|
|
|
for _, event := range feUpdate.Events {
|
2024-11-04 21:52:36 +01:00
|
|
|
if event.GlobalEventType != "" {
|
2024-11-02 18:58:13 +01:00
|
|
|
if impl.Client.GlobalEventHandler != nil {
|
|
|
|
impl.Client.GlobalEventHandler(impl.Client, event)
|
|
|
|
}
|
|
|
|
} else {
|
2024-11-04 21:52:36 +01:00
|
|
|
impl.Client.Root.Event(event.WaveId, event.EventType, event)
|
2024-11-02 18:58:13 +01:00
|
|
|
}
|
|
|
|
}
|
2024-11-06 08:07:45 +01:00
|
|
|
// update refs
|
|
|
|
for _, ref := range feUpdate.RefUpdates {
|
|
|
|
impl.Client.Root.UpdateRef(ref)
|
|
|
|
}
|
2024-11-06 00:52:59 +01:00
|
|
|
|
|
|
|
var update *vdom.VDomBackendUpdate
|
|
|
|
var err error
|
|
|
|
|
2024-11-04 21:52:36 +01:00
|
|
|
if feUpdate.Resync || true {
|
2024-11-06 00:52:59 +01:00
|
|
|
update, err = impl.Client.fullRender()
|
|
|
|
} else {
|
|
|
|
update, err = impl.Client.incrementalRender()
|
|
|
|
}
|
|
|
|
update.CreateTransferElems()
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
respChan <- wshrpc.RespOrErrorUnion[*vdom.VDomBackendUpdate]{
|
|
|
|
Error: err,
|
|
|
|
}
|
|
|
|
close(respChan)
|
|
|
|
return respChan
|
2024-11-02 18:58:13 +01:00
|
|
|
}
|
2024-11-06 00:52:59 +01:00
|
|
|
|
|
|
|
// Split the update into chunks and send them sequentially
|
|
|
|
updates := vdom.SplitBackendUpdate(update)
|
|
|
|
go func() {
|
|
|
|
defer close(respChan)
|
|
|
|
for _, splitUpdate := range updates {
|
|
|
|
respChan <- wshrpc.RespOrErrorUnion[*vdom.VDomBackendUpdate]{
|
|
|
|
Response: splitUpdate,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return respChan
|
2024-11-02 18:58:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (impl *VDomServerImpl) VDomUrlRequestCommand(ctx context.Context, data wshrpc.VDomUrlRequestData) chan wshrpc.RespOrErrorUnion[wshrpc.VDomUrlRequestResponse] {
|
|
|
|
respChan := make(chan wshrpc.RespOrErrorUnion[wshrpc.VDomUrlRequestResponse])
|
|
|
|
writer := NewStreamingResponseWriter(respChan)
|
|
|
|
|
|
|
|
go func() {
|
2024-11-06 00:52:59 +01:00
|
|
|
defer close(respChan) // Declared first, so it executes last
|
|
|
|
defer writer.Close() // Ensures writer is closed before the channel is closed
|
|
|
|
|
2024-11-02 18:58:13 +01:00
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
|
|
|
writer.WriteHeader(http.StatusInternalServerError)
|
|
|
|
writer.Write([]byte(fmt.Sprintf("internal server error: %v", r)))
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Create an HTTP request from the RPC request data
|
|
|
|
var bodyReader *bytes.Reader
|
|
|
|
if data.Body != nil {
|
|
|
|
bodyReader = bytes.NewReader(data.Body)
|
|
|
|
} else {
|
|
|
|
bodyReader = bytes.NewReader([]byte{})
|
|
|
|
}
|
|
|
|
|
|
|
|
httpReq, err := http.NewRequest(data.Method, data.URL, bodyReader)
|
|
|
|
if err != nil {
|
|
|
|
writer.WriteHeader(http.StatusInternalServerError)
|
|
|
|
writer.Write([]byte(err.Error()))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for key, value := range data.Headers {
|
|
|
|
httpReq.Header.Set(key, value)
|
|
|
|
}
|
2024-11-07 09:07:23 +01:00
|
|
|
if httpReq.URL.Path == "/wave/global.css" && impl.Client.GlobalStylesOption != nil {
|
|
|
|
ServeFileOption(writer, httpReq, *impl.Client.GlobalStylesOption)
|
|
|
|
return
|
|
|
|
}
|
2024-11-02 18:58:13 +01:00
|
|
|
if impl.Client.OverrideUrlHandler != nil {
|
|
|
|
impl.Client.OverrideUrlHandler.ServeHTTP(writer, httpReq)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
impl.Client.UrlHandlerMux.ServeHTTP(writer, httpReq)
|
|
|
|
}()
|
|
|
|
|
|
|
|
return respChan
|
|
|
|
}
|