Compare commits
6 commits
edad47b568
...
1810edd225
Author | SHA1 | Date | |
---|---|---|---|
![]() |
1810edd225 | ||
![]() |
1e3104c681 | ||
![]() |
3c46ab9416 | ||
![]() |
60635cce46 | ||
![]() |
b2cc3359a8 | ||
![]() |
54d958d480 |
11 changed files with 144 additions and 80 deletions
16
.forgejo/workflows/test.yml
Normal file
16
.forgejo/workflows/test.yml
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
on: [push]
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: docker
|
||||||
|
steps:
|
||||||
|
# Step 1: Check out the code
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
|
# Step 2: Set up go environment
|
||||||
|
- uses: actions/setup-go@v4
|
||||||
|
with:
|
||||||
|
go-version-file: '/workspace/FLUX/GoSBHPF/go.mod'
|
||||||
|
|
||||||
|
# Step 3: Set the working directory and run the project tests
|
||||||
|
- run: go test
|
||||||
|
working-directory: /workspace/FLUX/GoSBHPF/SBHPFv1
|
22
.gitignore
vendored
Normal file
22
.gitignore
vendored
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
# Binaries for programs and plugins
|
||||||
|
*.exe
|
||||||
|
*.exe~
|
||||||
|
*.dll
|
||||||
|
*.so
|
||||||
|
*.dylib
|
||||||
|
|
||||||
|
# Test binary, built with `go test -c`
|
||||||
|
*.test
|
||||||
|
|
||||||
|
# Output of the go coverage tool, specifically when used with LiteIDE
|
||||||
|
*.out
|
||||||
|
|
||||||
|
# Dependency directories (remove the comment below to include it)
|
||||||
|
# vendor/
|
||||||
|
|
||||||
|
# Go workspace file
|
||||||
|
go.work
|
||||||
|
go.work.sum
|
||||||
|
|
||||||
|
# env file
|
||||||
|
.env
|
10
.vscode/settings.json
vendored
Normal file
10
.vscode/settings.json
vendored
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
{
|
||||||
|
"cSpell.words": [
|
||||||
|
"SBHPF",
|
||||||
|
"sbhpfv",
|
||||||
|
"seekable",
|
||||||
|
"seekablebuffer",
|
||||||
|
"targetbytes",
|
||||||
|
"zervo"
|
||||||
|
]
|
||||||
|
}
|
|
@ -1,3 +1,6 @@
|
||||||
# GoSBHPF
|
# GoSBHPF
|
||||||
|
|
||||||
Go implementation of the SBHPF binary format.
|

|
||||||
|

|
||||||
|
|
||||||
|
GoSBHPF is a go library implementing the SBHPF binary format.
|
||||||
|
|
33
SBHPFv1/buffered_serializer.go
Normal file
33
SBHPFv1/buffered_serializer.go
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
package sbhpfv1
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"git.zervo.org/FLUX/GoSBHPF/pkg/seekablebuffer"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SerializeNodeBuffered serializes a node object into a SBHPF node.
|
||||||
|
// It is a wrapper around SerializeNode that temporarily stores the serialized nodes in a seekable memory buffer.
|
||||||
|
// This provides serialization compatibility with non-seekable writers, at the cost of increased memory usage.
|
||||||
|
func SerializeNodeBuffered(w io.Writer, node *Node) error {
|
||||||
|
buf := &seekablebuffer.Buffer{}
|
||||||
|
if err := SerializeNode(buf, node); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := io.Copy(w, buf)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// SerializeBuffered serializes a root node object into a complete SBHPF tree.
|
||||||
|
// It is a wrapper around Serialize that temporarily stores the serialized data in a seekable memory buffer.
|
||||||
|
// This provides serialization compatibility with non-seekable writers, at the cost of increased memory usage.
|
||||||
|
func SerializeBuffered(w io.Writer, node *Node) error {
|
||||||
|
buf := &seekablebuffer.Buffer{}
|
||||||
|
if err := Serialize(buf, node); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := io.Copy(w, buf)
|
||||||
|
return err
|
||||||
|
}
|
|
@ -2,7 +2,7 @@ package sbhpfv1
|
||||||
|
|
||||||
// Constants for file format.
|
// Constants for file format.
|
||||||
// These are statically set in version 1 of SBHPF,
|
// These are statically set in version 1 of SBHPF,
|
||||||
// but can be used to achieve an extended featureset in future versions,
|
// but can be used to achieve an extended feature-set in future versions,
|
||||||
// while maintaining backwards compatibility.
|
// while maintaining backwards compatibility.
|
||||||
const (
|
const (
|
||||||
FormatVersion = 0x01 // Current version
|
FormatVersion = 0x01 // Current version
|
||||||
|
|
|
@ -6,8 +6,9 @@ import (
|
||||||
"io"
|
"io"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Serialize writes a full binary property file from a root node object.
|
// Serialize serializes a root node object into a complete SBHPF tree.
|
||||||
// This includes the start headers necessary for proper deserialization of raw files.
|
// This includes the start headers necessary for proper deserialization of raw files.
|
||||||
|
// NOTE: Writer must support seeking. If this is not possible, use the SerializeBuffered wrapper.
|
||||||
func Serialize(w io.Writer, root *Node) error {
|
func Serialize(w io.Writer, root *Node) error {
|
||||||
// Write file header (version + feature flag).
|
// Write file header (version + feature flag).
|
||||||
if err := binary.Write(w, binary.LittleEndian, uint8(FormatVersion)); err != nil {
|
if err := binary.Write(w, binary.LittleEndian, uint8(FormatVersion)); err != nil {
|
||||||
|
@ -22,7 +23,7 @@ func Serialize(w io.Writer, root *Node) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
// SerializeNode serializes a node object into a SBHPF node.
|
// SerializeNode serializes a node object into a SBHPF node.
|
||||||
// NOTE: Writer must support seeking. If this is not possible, use SerializeNodeStream.
|
// NOTE: Writer must support seeking. If this is not possible, use the SerializeNodeBuffered wrapper.
|
||||||
func SerializeNode(w io.Writer, node *Node) error {
|
func SerializeNode(w io.Writer, node *Node) error {
|
||||||
// Reserve space for node size ("Node Size" node header).
|
// Reserve space for node size ("Node Size" node header).
|
||||||
sizePos := getWriterPosition(w) // Save current position.
|
sizePos := getWriterPosition(w) // Save current position.
|
||||||
|
|
|
@ -18,12 +18,14 @@ var (
|
||||||
}
|
}
|
||||||
|
|
||||||
node_ser_targetbytes = []byte{
|
node_ser_targetbytes = []byte{
|
||||||
// ROOT NODE
|
// -- ROOT NODE: HEADER
|
||||||
0x33, 0x00, 0x00, 0x00, // Node size = 33
|
0x33, 0x00, 0x00, 0x00, // Node size = 33
|
||||||
0x02, 0x00, // Property count = 2
|
0x02, 0x00, // Property count = 2
|
||||||
0x01, 0x00, // Child count = 1
|
0x01, 0x00, // Child count = 1
|
||||||
0x06, // Name length = 6
|
0x06, // Name length = 6
|
||||||
|
// -- ROOT NODE: NAME
|
||||||
0x70, 0x6c, 0x61, 0x79, 0x65, 0x72, // Node name = "player"
|
0x70, 0x6c, 0x61, 0x79, 0x65, 0x72, // Node name = "player"
|
||||||
|
// -- ROOT NODE: PROPERTIES
|
||||||
// ROOT NODE -> PROPERTY A
|
// ROOT NODE -> PROPERTY A
|
||||||
0x06, // Key length = 6
|
0x06, // Key length = 6
|
||||||
0x0b, // Value type = bool
|
0x0b, // Value type = bool
|
||||||
|
@ -34,6 +36,7 @@ var (
|
||||||
0x03, // Value type = 16-bit signed int
|
0x03, // Value type = 16-bit signed int
|
||||||
0x6c, 0x65, 0x76, 0x65, 0x6c, // Key = "level"
|
0x6c, 0x65, 0x76, 0x65, 0x6c, // Key = "level"
|
||||||
0x1b, 0x00, // Value = 27
|
0x1b, 0x00, // Value = 27
|
||||||
|
// -- ROOT NODE: CHILDREN
|
||||||
// ROOT NODE -> CHILD NODE
|
// ROOT NODE -> CHILD NODE
|
||||||
0x12, 0x00, 0x00, 0x00, // Node size = 12
|
0x12, 0x00, 0x00, 0x00, // Node size = 12
|
||||||
0x00, 0x00, // Property count = 0
|
0x00, 0x00, // Property count = 0
|
||||||
|
@ -41,6 +44,25 @@ var (
|
||||||
0x09, // Name length = 9
|
0x09, // Name length = 9
|
||||||
0x69, 0x6e, 0x76, 0x65, 0x6e, 0x74, 0x6f, 0x72, 0x79, // Node name = "inventory"
|
0x69, 0x6e, 0x76, 0x65, 0x6e, 0x74, 0x6f, 0x72, 0x79, // Node name = "inventory"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
full_ser_targetbytes = []byte{
|
||||||
|
// -- FILE HEADER
|
||||||
|
0x01, // Version flag = 1
|
||||||
|
0x00, // Feature flag = 0
|
||||||
|
// -- ROOT NODE: HEADER
|
||||||
|
0x16, 0x00, 0x00, 0x00, // Node size = 16
|
||||||
|
0x01, 0x00, // Property count = 1
|
||||||
|
0x00, 0x00, // Child count = 0
|
||||||
|
0x04, // Name length = 4
|
||||||
|
// -- ROOT NODE: NAME
|
||||||
|
0x75, 0x73, 0x65, 0x72, // Node name = "user"
|
||||||
|
// -- ROOT NODE: PROPERTIES
|
||||||
|
// ROOT NODE -> PROPERTY
|
||||||
|
0x06, // Key length = 6
|
||||||
|
0x0b, // Value type = bool
|
||||||
|
0x61, 0x63, 0x74, 0x69, 0x76, 0x65, // Key = "active"
|
||||||
|
0x01, // Value = true
|
||||||
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestPropertySerialization(t *testing.T) {
|
func TestPropertySerialization(t *testing.T) {
|
||||||
|
@ -50,8 +72,8 @@ func TestPropertySerialization(t *testing.T) {
|
||||||
Value: "zervo",
|
Value: "zervo",
|
||||||
}
|
}
|
||||||
|
|
||||||
buf := &bytes.Buffer{}
|
var buf bytes.Buffer
|
||||||
err := sbhpfv1.SerializeProperty(buf, prop)
|
err := sbhpfv1.SerializeProperty(&buf, prop)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("Property serialization failed: %v", err)
|
t.Fatalf("Property serialization failed: %v", err)
|
||||||
}
|
}
|
||||||
|
@ -89,13 +111,38 @@ func TestNodeSerialization(t *testing.T) {
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
var w bytes.Buffer
|
var buf bytes.Buffer
|
||||||
err := sbhpfv1.SerializeNodeStream(&w, &node)
|
err := sbhpfv1.SerializeNodeBuffered(&buf, &node)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("Node serialization failed: %v", err)
|
t.Fatalf("Node serialization failed: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if !slices.Equal(w.Bytes(), node_ser_targetbytes) {
|
if !slices.Equal(buf.Bytes(), node_ser_targetbytes) {
|
||||||
t.Fatalf("Node serialization generated bad data: %v", w.Bytes())
|
t.Fatalf("Node serialization generated bad data: %v", buf.Bytes())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFullSerialization(t *testing.T) {
|
||||||
|
prop := sbhpfv1.Property{
|
||||||
|
Key: "active",
|
||||||
|
Type: sbhpfv1.TypeBool,
|
||||||
|
Value: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
node := sbhpfv1.Node{
|
||||||
|
Name: "user",
|
||||||
|
Properties: []sbhpfv1.Property{
|
||||||
|
prop,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err := sbhpfv1.SerializeBuffered(&buf, &node)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Full tree serialization failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !slices.Equal(buf.Bytes(), full_ser_targetbytes) {
|
||||||
|
t.Fatalf("Full tree serialization generated bad data: %v", buf.Bytes())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,20 +0,0 @@
|
||||||
package sbhpfv1
|
|
||||||
|
|
||||||
import (
|
|
||||||
"io"
|
|
||||||
|
|
||||||
"git.zervo.org/FLUX/GoSBHPF/pkg/seekablebuffer"
|
|
||||||
)
|
|
||||||
|
|
||||||
// SerializeNodeStream serializes a node object into a SBHPF node.
|
|
||||||
// It is a wrapper around SerializeNode that temporarily stores the serialized nodes in a seekable memory buffer.
|
|
||||||
// This is less efficient, but provides compatibility with writers that lack seeking support.
|
|
||||||
func SerializeNodeStream(w io.Writer, node *Node) error {
|
|
||||||
buf := &seekablebuffer.Buffer{}
|
|
||||||
if err := SerializeNode(buf, node); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
_, err := io.Copy(w, buf)
|
|
||||||
return err
|
|
||||||
}
|
|
48
main.go
48
main.go
|
@ -1,48 +0,0 @@
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
sbhpfv1 "git.zervo.org/FLUX/GoSBHPF/SBHPFv1"
|
|
||||||
)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
file, err := os.Create("test.bin")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
defer file.Close()
|
|
||||||
|
|
||||||
prop_a := sbhpfv1.Property{
|
|
||||||
Key: "active",
|
|
||||||
Type: sbhpfv1.TypeBool,
|
|
||||||
Value: true,
|
|
||||||
}
|
|
||||||
|
|
||||||
prop_b := sbhpfv1.Property{
|
|
||||||
Key: "level",
|
|
||||||
Type: sbhpfv1.TypeInt16,
|
|
||||||
Value: int16(27),
|
|
||||||
}
|
|
||||||
|
|
||||||
child_node := sbhpfv1.Node{
|
|
||||||
Name: "inventory",
|
|
||||||
}
|
|
||||||
|
|
||||||
node := sbhpfv1.Node{
|
|
||||||
Name: "player",
|
|
||||||
Properties: []sbhpfv1.Property{
|
|
||||||
prop_a,
|
|
||||||
prop_b,
|
|
||||||
},
|
|
||||||
Children: []*sbhpfv1.Node{
|
|
||||||
&child_node,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
err = sbhpfv1.SerializeNode(file, &node)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
}
|
|
BIN
test.bin
BIN
test.bin
Binary file not shown.
Loading…
Add table
Reference in a new issue