mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-23 21:51:30 +01:00
a639d72e30
* take a stab at fixing #99. ignore invalid output before we see a real packet. the complication here was ensuring we always output a real packet in every flow so we don't actually lose valid errors. * add ping packets to prime the parser (when in ignoreUntilValid mode)
255 lines
5.1 KiB
Go
255 lines
5.1 KiB
Go
// Copyright 2023, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package packet
|
|
|
|
import (
|
|
"bufio"
|
|
"context"
|
|
"io"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
)
|
|
|
|
type PacketParser struct {
|
|
Lock *sync.Mutex
|
|
MainCh chan PacketType
|
|
RpcMap map[string]*RpcEntry
|
|
RpcHandler bool
|
|
Err error
|
|
}
|
|
|
|
type RpcEntry struct {
|
|
ReqId string
|
|
RespCh chan RpcResponsePacketType
|
|
}
|
|
|
|
type RpcResponseIter struct {
|
|
ReqId string
|
|
Parser *PacketParser
|
|
}
|
|
|
|
func (iter *RpcResponseIter) Next(ctx context.Context) (RpcResponsePacketType, error) {
|
|
// will unregister the rpc on ResponseDone
|
|
return iter.Parser.GetNextResponse(ctx, iter.ReqId)
|
|
}
|
|
|
|
func (iter *RpcResponseIter) Close() {
|
|
iter.Parser.UnRegisterRpc(iter.ReqId)
|
|
}
|
|
|
|
func CombinePacketParsers(p1 *PacketParser, p2 *PacketParser, rpcHandler bool) *PacketParser {
|
|
rtnParser := &PacketParser{
|
|
Lock: &sync.Mutex{},
|
|
MainCh: make(chan PacketType),
|
|
RpcMap: make(map[string]*RpcEntry),
|
|
RpcHandler: rpcHandler,
|
|
}
|
|
var wg sync.WaitGroup
|
|
wg.Add(2)
|
|
go func() {
|
|
defer wg.Done()
|
|
for pk := range p1.MainCh {
|
|
if rtnParser.RpcHandler {
|
|
sent := rtnParser.trySendRpcResponse(pk)
|
|
if sent {
|
|
continue
|
|
}
|
|
}
|
|
rtnParser.MainCh <- pk
|
|
}
|
|
}()
|
|
go func() {
|
|
defer wg.Done()
|
|
for pk := range p2.MainCh {
|
|
if rtnParser.RpcHandler {
|
|
sent := rtnParser.trySendRpcResponse(pk)
|
|
if sent {
|
|
continue
|
|
}
|
|
}
|
|
rtnParser.MainCh <- pk
|
|
}
|
|
}()
|
|
go func() {
|
|
wg.Wait()
|
|
close(rtnParser.MainCh)
|
|
}()
|
|
return rtnParser
|
|
}
|
|
|
|
// should have already registered rpc
|
|
func (p *PacketParser) WaitForResponse(ctx context.Context, reqId string) RpcResponsePacketType {
|
|
entry := p.getRpcEntry(reqId)
|
|
if entry == nil {
|
|
return nil
|
|
}
|
|
defer p.UnRegisterRpc(reqId)
|
|
select {
|
|
case resp := <-entry.RespCh:
|
|
return resp
|
|
case <-ctx.Done():
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (p *PacketParser) GetResponseIter(reqId string) *RpcResponseIter {
|
|
return &RpcResponseIter{Parser: p, ReqId: reqId}
|
|
}
|
|
|
|
func (p *PacketParser) GetNextResponse(ctx context.Context, reqId string) (RpcResponsePacketType, error) {
|
|
entry := p.getRpcEntry(reqId)
|
|
if entry == nil {
|
|
return nil, nil
|
|
}
|
|
select {
|
|
case resp := <-entry.RespCh:
|
|
if resp.GetResponseDone() {
|
|
p.UnRegisterRpc(reqId)
|
|
}
|
|
return resp, nil
|
|
case <-ctx.Done():
|
|
return nil, ctx.Err()
|
|
}
|
|
}
|
|
|
|
func (p *PacketParser) UnRegisterRpc(reqId string) {
|
|
p.Lock.Lock()
|
|
defer p.Lock.Unlock()
|
|
entry := p.RpcMap[reqId]
|
|
if entry != nil {
|
|
close(entry.RespCh)
|
|
delete(p.RpcMap, reqId)
|
|
}
|
|
}
|
|
|
|
func (p *PacketParser) RegisterRpc(reqId string) chan RpcResponsePacketType {
|
|
return p.RegisterRpcSz(reqId, 2)
|
|
}
|
|
|
|
func (p *PacketParser) RegisterRpcSz(reqId string, queueSize int) chan RpcResponsePacketType {
|
|
p.Lock.Lock()
|
|
defer p.Lock.Unlock()
|
|
ch := make(chan RpcResponsePacketType, queueSize)
|
|
entry := &RpcEntry{ReqId: reqId, RespCh: ch}
|
|
p.RpcMap[reqId] = entry
|
|
return ch
|
|
}
|
|
|
|
func (p *PacketParser) getRpcEntry(reqId string) *RpcEntry {
|
|
p.Lock.Lock()
|
|
defer p.Lock.Unlock()
|
|
entry := p.RpcMap[reqId]
|
|
return entry
|
|
}
|
|
|
|
func (p *PacketParser) trySendRpcResponse(pk PacketType) bool {
|
|
respPk, ok := pk.(RpcResponsePacketType)
|
|
if !ok {
|
|
return false
|
|
}
|
|
p.Lock.Lock()
|
|
defer p.Lock.Unlock()
|
|
entry := p.RpcMap[respPk.GetResponseId()]
|
|
if entry == nil {
|
|
return false
|
|
}
|
|
// nonblocking send
|
|
select {
|
|
case entry.RespCh <- respPk:
|
|
default:
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (p *PacketParser) GetErr() error {
|
|
p.Lock.Lock()
|
|
defer p.Lock.Unlock()
|
|
return p.Err
|
|
}
|
|
|
|
func (p *PacketParser) SetErr(err error) {
|
|
p.Lock.Lock()
|
|
defer p.Lock.Unlock()
|
|
if p.Err == nil {
|
|
p.Err = err
|
|
}
|
|
}
|
|
|
|
type PacketParserOpts struct {
|
|
RpcHandler bool
|
|
IgnoreUntilValid bool
|
|
}
|
|
|
|
func MakePacketParser(input io.Reader, opts *PacketParserOpts) *PacketParser {
|
|
if opts == nil {
|
|
opts = &PacketParserOpts{}
|
|
}
|
|
parser := &PacketParser{
|
|
Lock: &sync.Mutex{},
|
|
MainCh: make(chan PacketType),
|
|
RpcMap: make(map[string]*RpcEntry),
|
|
RpcHandler: opts.RpcHandler,
|
|
}
|
|
ignoreUntilValid := opts.IgnoreUntilValid
|
|
bufReader := bufio.NewReader(input)
|
|
go func() {
|
|
defer func() {
|
|
close(parser.MainCh)
|
|
}()
|
|
for {
|
|
line, err := bufReader.ReadString('\n')
|
|
if err == io.EOF {
|
|
return
|
|
}
|
|
if err != nil {
|
|
parser.SetErr(err)
|
|
return
|
|
}
|
|
if line == "\n" {
|
|
continue
|
|
}
|
|
// ##[len][json]\n
|
|
// ##14{"hello":true}\n
|
|
// ##N{...}
|
|
hasPrefix := strings.HasPrefix(line, "##")
|
|
bracePos := strings.Index(line, "{")
|
|
if !hasPrefix || bracePos == -1 {
|
|
if !ignoreUntilValid {
|
|
parser.MainCh <- MakeRawPacket(line[:len(line)-1])
|
|
}
|
|
continue
|
|
}
|
|
ignoreUntilValid = false
|
|
packetLen := -1
|
|
if line[2:bracePos] != "N" {
|
|
packetLen, err = strconv.Atoi(line[2:bracePos])
|
|
if err != nil || packetLen != len(line)-bracePos-1 {
|
|
parser.MainCh <- MakeRawPacket(line[:len(line)-1])
|
|
continue
|
|
}
|
|
}
|
|
pk, err := ParseJsonPacket([]byte(line[bracePos:]))
|
|
if err != nil {
|
|
parser.MainCh <- MakeRawPacket(line[:len(line)-1])
|
|
continue
|
|
}
|
|
if pk.GetType() == DonePacketStr {
|
|
return
|
|
}
|
|
if pk.GetType() == PingPacketStr {
|
|
continue
|
|
}
|
|
if parser.RpcHandler {
|
|
sent := parser.trySendRpcResponse(pk)
|
|
if sent {
|
|
continue
|
|
}
|
|
}
|
|
parser.MainCh <- pk
|
|
}
|
|
}()
|
|
return parser
|
|
}
|