Skip to content

Commit 6662a21

Browse files
Jilleklauspost
andauthored
s2: Document and test how to peek the stream for skippable blocks (#918)
Co-authored-by: Klaus Post <[email protected]>
1 parent 3deb878 commit 6662a21

File tree

2 files changed

+49
-0
lines changed

2 files changed

+49
-0
lines changed

s2/reader.go

+4
Original file line numberDiff line numberDiff line change
@@ -104,6 +104,8 @@ func ReaderIgnoreStreamIdentifier() ReaderOption {
104104
// For each chunk with the ID, the callback is called with the content.
105105
// Any returned non-nil error will abort decompression.
106106
// Only one callback per ID is supported, latest sent will be used.
107+
// You can peek the stream, triggering the callback, by doing a Read with a 0
108+
// byte buffer.
107109
func ReaderSkippableCB(id uint8, fn func(r io.Reader) error) ReaderOption {
108110
return func(r *Reader) error {
109111
if id < 0x80 || id > 0xfd {
@@ -1053,6 +1055,8 @@ func (r *Reader) ReadByte() (byte, error) {
10531055
// Any returned non-nil error will abort decompression.
10541056
// Only one callback per ID is supported, latest sent will be used.
10551057
// Sending a nil function will disable previous callbacks.
1058+
// You can peek the stream, triggering the callback, by doing a Read with a 0
1059+
// byte buffer.
10561060
func (r *Reader) SkippableCB(id uint8, fn func(r io.Reader) error) error {
10571061
if id < 0x80 || id >= chunkTypePadding {
10581062
return fmt.Errorf("ReaderSkippableCB: Invalid id provided, must be 0x80-0xfe (inclusive)")

s2/reader_test.go

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
// Copyright (c) 2019+ Klaus Post. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package s2
6+
7+
import (
8+
"bytes"
9+
"io"
10+
"testing"
11+
)
12+
13+
func TestLeadingSkippableBlock(t *testing.T) {
14+
var buf bytes.Buffer
15+
w := NewWriter(&buf)
16+
if err := w.AddSkippableBlock(0x80, []byte("skippable block")); err != nil {
17+
t.Fatalf("w.AddSkippableBlock: %v", err)
18+
}
19+
if _, err := w.Write([]byte("some data")); err != nil {
20+
t.Fatalf("w.Write: %v", err)
21+
}
22+
if err := w.Close(); err != nil {
23+
t.Fatalf("w.Close: %v", err)
24+
}
25+
r := NewReader(&buf)
26+
var sb []byte
27+
r.SkippableCB(0x80, func(sr io.Reader) error {
28+
var err error
29+
sb, err = io.ReadAll(sr)
30+
return err
31+
})
32+
if _, err := r.Read([]byte{}); err != nil {
33+
t.Errorf("empty read failed: %v", err)
34+
}
35+
if !bytes.Equal(sb, []byte("skippable block")) {
36+
t.Errorf("didn't get correct data from skippable block: %q", string(sb))
37+
}
38+
data, err := io.ReadAll(r)
39+
if err != nil {
40+
t.Fatalf("r.Read: %v", err)
41+
}
42+
if !bytes.Equal(data, []byte("some data")) {
43+
t.Errorf("didn't get correct compressed data: %q", string(data))
44+
}
45+
}

0 commit comments

Comments
 (0)