-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcompression.go
94 lines (78 loc) · 1.75 KB
/
compression.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
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
package main
import (
"compress/gzip"
"context"
"fmt"
"io"
"sync"
"github.com/klauspost/compress/zstd"
)
type CompressionAlgorithm = string
const (
Zstd = "zstd"
Gzip = "gzip"
)
type compression struct {
compressionReader io.ReadCloser
err error
wg *sync.WaitGroup
finished chan struct{}
}
func (c *compression) Read(b []byte) (int, error) {
n, err := c.compressionReader.Read(b)
if c.err != nil {
return n, c.err
}
return n, err
}
func (c *compression) Close() error {
err := c.compressionReader.Close()
c.wg.Wait()
if c.err != nil {
return c.err
}
return err
}
func newCompressionReader(ctx context.Context, reader io.ReadCloser, level int, algo CompressionAlgorithm) (io.ReadCloser, <-chan struct{}, error) {
c := &compression{}
readerPipe, writePipe := io.Pipe()
w, err := func() (io.WriteCloser, error) {
switch algo {
case Zstd:
w, err := zstd.NewWriter(writePipe, zstd.WithEncoderLevel(zstd.EncoderLevel(level)))
if err != nil {
return nil, fmt.Errorf("zstd init write level: %w", err)
}
return w, nil
case Gzip:
w, err := gzip.NewWriterLevel(writePipe, level)
if err != nil {
return nil, fmt.Errorf("gzip init write level: %w", err)
}
return w, nil
default:
return nil, fmt.Errorf("not implemented %q", algo)
}
}()
if err != nil {
return nil, nil, fmt.Errorf("get compression writer: %w", err)
}
wg := &sync.WaitGroup{}
wg.Add(1)
var readerGzip = reader
c.finished = make(chan struct{}, 1)
go func() {
defer wg.Done()
_, err := io.Copy(w, readerGzip)
if err != nil {
c.err = err
}
w.Close()
readerGzip.Close()
writePipe.Close()
close(c.finished)
}()
c.compressionReader = readerPipe
c.wg = wg
return c, c.finished, nil
}