mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-07 19:28:44 +01:00
a1e4e807cc
* remote statemap from waveshell server (diff against initial state) * move ShellStatePtr from sstore to packet so it can be passed over the wire * add finalstatebaseptr to cmddone * much improved diff computation code on wavesrv side * fix displayname -- now using hash * add comments, change a couple msh.WriteToPtyBuffer calls to log.Printfs
210 lines
5.2 KiB
Go
210 lines
5.2 KiB
Go
// Copyright 2023, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package statediff
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"slices"
|
|
|
|
"github.com/wavetermdev/waveterm/waveshell/pkg/binpack"
|
|
"github.com/wavetermdev/waveterm/waveshell/pkg/utilfn"
|
|
)
|
|
|
|
const MapDiffVersion_0 = 0
|
|
const MapDiffVersion = 1
|
|
|
|
// 0-bytes are not allowed in entries or keys (same as bash)
|
|
|
|
type MapDiffType struct {
|
|
ToAdd map[string][]byte
|
|
ToRemove []string
|
|
}
|
|
|
|
func (diff *MapDiffType) Clear() {
|
|
diff.ToAdd = nil
|
|
diff.ToRemove = nil
|
|
}
|
|
|
|
func (diff MapDiffType) Dump() {
|
|
fmt.Printf("VAR-DIFF +%d -%d\n", len(diff.ToAdd), len(diff.ToRemove))
|
|
for name, val := range diff.ToAdd {
|
|
fmt.Printf(" add[%s] %s\n", name, val)
|
|
}
|
|
for _, name := range diff.ToRemove {
|
|
fmt.Printf(" rem[%s]\n", name)
|
|
}
|
|
}
|
|
|
|
func makeMapDiff(oldMap map[string][]byte, newMap map[string][]byte) MapDiffType {
|
|
var rtn MapDiffType
|
|
rtn.ToAdd = make(map[string][]byte)
|
|
for name, newVal := range newMap {
|
|
oldVal, found := oldMap[name]
|
|
if !found || !bytes.Equal(oldVal, newVal) {
|
|
rtn.ToAdd[name] = newVal
|
|
continue
|
|
}
|
|
}
|
|
for name := range oldMap {
|
|
_, found := newMap[name]
|
|
if !found {
|
|
rtn.ToRemove = append(rtn.ToRemove, name)
|
|
}
|
|
}
|
|
return rtn
|
|
}
|
|
|
|
func (diff MapDiffType) apply(oldMap map[string][]byte) map[string][]byte {
|
|
rtn := make(map[string][]byte)
|
|
for name, val := range oldMap {
|
|
rtn[name] = val
|
|
}
|
|
for name, val := range diff.ToAdd {
|
|
rtn[name] = val
|
|
}
|
|
for _, name := range diff.ToRemove {
|
|
delete(rtn, name)
|
|
}
|
|
return rtn
|
|
}
|
|
|
|
// this is kept for reference
|
|
func (diff MapDiffType) Encode_v0() []byte {
|
|
var buf bytes.Buffer
|
|
viBuf := make([]byte, binary.MaxVarintLen64)
|
|
putUVarint(&buf, viBuf, MapDiffVersion_0)
|
|
putUVarint(&buf, viBuf, len(diff.ToAdd))
|
|
for key, val := range diff.ToAdd {
|
|
buf.WriteString(key)
|
|
buf.WriteByte(0)
|
|
buf.Write(val)
|
|
buf.WriteByte(0)
|
|
}
|
|
for _, val := range diff.ToRemove {
|
|
buf.WriteString(val)
|
|
buf.WriteByte(0)
|
|
}
|
|
return buf.Bytes()
|
|
}
|
|
|
|
// we sort map keys and remove values to make the diff deterministic
|
|
func (diff MapDiffType) Encode() []byte {
|
|
var buf bytes.Buffer
|
|
binpack.PackUInt(&buf, MapDiffVersion)
|
|
binpack.PackUInt(&buf, uint64(len(diff.ToAdd)))
|
|
addKeys := utilfn.GetOrderedMapKeys(diff.ToAdd)
|
|
for _, key := range addKeys {
|
|
val := diff.ToAdd[key]
|
|
binpack.PackValue(&buf, []byte(key))
|
|
binpack.PackValue(&buf, val)
|
|
}
|
|
slices.Sort(diff.ToRemove)
|
|
binpack.PackUInt(&buf, uint64(len(diff.ToRemove)))
|
|
for _, val := range diff.ToRemove {
|
|
binpack.PackValue(&buf, []byte(val))
|
|
}
|
|
return buf.Bytes()
|
|
}
|
|
|
|
func (diff *MapDiffType) Decode(diffBytes []byte) error {
|
|
diff.Clear()
|
|
if len(diffBytes) == 0 {
|
|
return nil
|
|
}
|
|
r := bytes.NewBuffer(diffBytes)
|
|
version, err := binpack.UnpackUInt(r)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid diff, cannot read version: %v", err)
|
|
}
|
|
if version == MapDiffVersion_0 {
|
|
return diff.Decode_v0(diffBytes)
|
|
}
|
|
if version != MapDiffVersion {
|
|
return fmt.Errorf("invalid diff, bad version: %d", version)
|
|
}
|
|
addLen, err := binpack.UnpackUIntAsInt(r)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid diff, cannot read add length: %v", err)
|
|
}
|
|
diff.ToAdd = make(map[string][]byte)
|
|
for i := 0; i < addLen; i++ {
|
|
key, err := binpack.UnpackValue(r)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid diff, cannot read add key %d: %v", i, err)
|
|
}
|
|
val, err := binpack.UnpackValue(r)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid diff, cannot read add val %d: %v", i, err)
|
|
}
|
|
diff.ToAdd[string(key)] = val
|
|
}
|
|
removeLen, err := binpack.UnpackUIntAsInt(r)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid diff, cannot read remove length: %v", err)
|
|
}
|
|
for i := 0; i < removeLen; i++ {
|
|
val, err := binpack.UnpackValue(r)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid diff, cannot read remove val %d: %v", i, err)
|
|
}
|
|
diff.ToRemove = append(diff.ToRemove, string(val))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (diff *MapDiffType) Decode_v0(diffBytes []byte) error {
|
|
r := bytes.NewBuffer(diffBytes)
|
|
version, err := binary.ReadUvarint(r)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid diff, cannot read version: %v", err)
|
|
}
|
|
if version != MapDiffVersion_0 {
|
|
return fmt.Errorf("invalid diff, bad version: %d", version)
|
|
}
|
|
mapLen64, err := binary.ReadUvarint(r)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid diff, cannot map length: %v", err)
|
|
}
|
|
mapLen := int(mapLen64)
|
|
fields := bytes.Split(r.Bytes(), []byte{0})
|
|
if len(fields) < 2*mapLen {
|
|
return fmt.Errorf("invalid diff, not enough fields, maplen:%d fields:%d", mapLen, len(fields))
|
|
}
|
|
mapFields := fields[0 : 2*mapLen]
|
|
removeFields := fields[2*mapLen:]
|
|
diff.ToAdd = make(map[string][]byte)
|
|
for i := 0; i < len(mapFields); i += 2 {
|
|
diff.ToAdd[string(mapFields[i])] = mapFields[i+1]
|
|
}
|
|
for _, removeVal := range removeFields {
|
|
if len(removeVal) == 0 {
|
|
continue
|
|
}
|
|
diff.ToRemove = append(diff.ToRemove, string(removeVal))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func MakeMapDiff(m1 map[string][]byte, m2 map[string][]byte) []byte {
|
|
diff := makeMapDiff(m1, m2)
|
|
if len(diff.ToAdd) == 0 && len(diff.ToRemove) == 0 {
|
|
return nil
|
|
}
|
|
return diff.Encode()
|
|
}
|
|
|
|
func ApplyMapDiff(oldMap map[string][]byte, diffBytes []byte) (map[string][]byte, error) {
|
|
if len(diffBytes) == 0 {
|
|
return oldMap, nil
|
|
}
|
|
var diff MapDiffType
|
|
err := diff.Decode(diffBytes)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return diff.apply(oldMap), nil
|
|
}
|