Add SeekableBuffer
Add a specialised bytes.Buffer variant with seeking capabilities. This allows writers without seeking support to serialize using NodeSerializerStreamed.
This commit is contained in:
parent
b7f8103a94
commit
96f0b64a11
2 changed files with 95 additions and 4 deletions
|
@ -1,15 +1,16 @@
|
||||||
package sbhpfv1
|
package sbhpfv1
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
|
"git.zervo.org/FLUX/GoSBHPF/pkg/seekablebuffer"
|
||||||
)
|
)
|
||||||
|
|
||||||
// SerializeNodeStream serializes a node object into a SBHPF node.
|
// SerializeNodeStream serializes a node object into a SBHPF node.
|
||||||
// It is a wrapper around SerializeNode that temporarily stores the serialized nodes in a memory buffer.
|
// It is a wrapper around SerializeNode that temporarily stores the serialized nodes in a seekable memory buffer.
|
||||||
// This is less efficient, but also works with writers that lack seeking support.
|
// This is less efficient, but provides compatibility with writers that lack seeking support.
|
||||||
func SerializeNodeStream(w io.Writer, node *Node) error {
|
func SerializeNodeStream(w io.Writer, node *Node) error {
|
||||||
buf := &bytes.Buffer{}
|
buf := &seekablebuffer.Buffer{}
|
||||||
if err := SerializeNode(buf, node); err != nil {
|
if err := SerializeNode(buf, node); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
90
pkg/seekablebuffer/seekablebuffer.go
Normal file
90
pkg/seekablebuffer/seekablebuffer.go
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
/*
|
||||||
|
SeekableBuffer
|
||||||
|
|
||||||
|
Copyright (c) 2023 aler9
|
||||||
|
Copyright (c) 2025 Zervó Zadachin
|
||||||
|
|
||||||
|
Implementation based on code from "bluenviron/mediacommon".
|
||||||
|
See: https://github.com/bluenviron/mediacommon/blob/main/pkg/formats/fmp4/seekablebuffer/seekablebuffer.go
|
||||||
|
|
||||||
|
Original code was licensed under the MIT License.
|
||||||
|
See: https://github.com/bluenviron/mediacommon/blob/main/LICENSE
|
||||||
|
*/
|
||||||
|
|
||||||
|
package seekablebuffer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Buffer is a bytes.Buffer with an additional Seek() method.
|
||||||
|
type Buffer struct {
|
||||||
|
bytes.Buffer
|
||||||
|
pos int64
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write implements io.Writer.
|
||||||
|
func (b *Buffer) Write(p []byte) (int, error) {
|
||||||
|
var n int
|
||||||
|
|
||||||
|
if b.pos < int64(b.Len()) {
|
||||||
|
n = copy(b.Bytes()[b.pos:], p)
|
||||||
|
p = p[n:]
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(p) > 0 {
|
||||||
|
// Buffer.Write should never return an error here.
|
||||||
|
nn, _ := b.Buffer.Write(p)
|
||||||
|
n += nn
|
||||||
|
}
|
||||||
|
|
||||||
|
b.pos += int64(n)
|
||||||
|
return n, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read (almost) implements io.Reader.
|
||||||
|
func (b *Buffer) Read(_ []byte) (int, error) {
|
||||||
|
return 0, fmt.Errorf("read not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Seek implements io.Seeker.
|
||||||
|
func (b *Buffer) Seek(offset int64, whence int) (int64, error) {
|
||||||
|
var pos2 int64
|
||||||
|
|
||||||
|
switch whence {
|
||||||
|
case io.SeekStart:
|
||||||
|
pos2 = offset
|
||||||
|
|
||||||
|
case io.SeekCurrent:
|
||||||
|
pos2 = b.pos + offset
|
||||||
|
|
||||||
|
case io.SeekEnd:
|
||||||
|
pos2 = int64(b.Len()) + offset
|
||||||
|
|
||||||
|
default:
|
||||||
|
return 0, fmt.Errorf("invalid seek whence")
|
||||||
|
}
|
||||||
|
|
||||||
|
if pos2 < 0 {
|
||||||
|
return 0, fmt.Errorf("negative position")
|
||||||
|
}
|
||||||
|
|
||||||
|
b.pos = pos2
|
||||||
|
|
||||||
|
// Expand buffer if needed
|
||||||
|
diff := b.pos - int64(b.Len())
|
||||||
|
if diff > 0 {
|
||||||
|
// Buffer.Write should never return an error here.
|
||||||
|
b.Buffer.Write(make([]byte, diff))
|
||||||
|
}
|
||||||
|
|
||||||
|
return pos2, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset resets the buffer state.
|
||||||
|
func (b *Buffer) Reset() {
|
||||||
|
b.Buffer.Reset()
|
||||||
|
b.pos = 0
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue