-
Notifications
You must be signed in to change notification settings - Fork 741
/
Copy pathzstd_compressor.go
58 lines (48 loc) · 1.58 KB
/
zstd_compressor.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
// Copyright (C) 2019-2024, Ava Labs, Inc. All rights reserved.
// See the file LICENSE for licensing terms.
package compression
import (
"bytes"
"fmt"
"io"
"math"
"github.com/DataDog/zstd"
)
var _ Compressor = (*zstdCompressor)(nil)
func NewZstdCompressor(maxSize int64) (Compressor, error) {
if maxSize == math.MaxInt64 {
// "Decompress" creates "io.LimitReader" with max size + 1:
// if the max size + 1 overflows, "io.LimitReader" reads nothing
// returning 0 byte for the decompress call
// require max size < math.MaxInt64 to prevent int64 overflows
return nil, ErrInvalidMaxSizeCompressor
}
return &zstdCompressor{
maxSize: maxSize,
}, nil
}
type zstdCompressor struct {
maxSize int64
}
func (z *zstdCompressor) Compress(msg []byte) ([]byte, error) {
if int64(len(msg)) > z.maxSize {
return nil, fmt.Errorf("%w: (%d) > (%d)", ErrMsgTooLarge, len(msg), z.maxSize)
}
return zstd.Compress(nil, msg)
}
func (z *zstdCompressor) Decompress(msg []byte) ([]byte, error) {
reader := zstd.NewReader(bytes.NewReader(msg))
defer reader.Close()
// We allow [io.LimitReader] to read up to [z.maxSize + 1] bytes, so that if
// the decompressed payload is greater than the maximum size, this function
// will return the appropriate error instead of an incomplete byte slice.
limitReader := io.LimitReader(reader, z.maxSize+1)
decompressed, err := io.ReadAll(limitReader)
if err != nil {
return nil, err
}
if int64(len(decompressed)) > z.maxSize {
return nil, fmt.Errorf("%w: (%d) > (%d)", ErrDecompressedMsgTooLarge, len(decompressed), z.maxSize)
}
return decompressed, nil
}