1package snappy
2
3import (
4	"bytes"
5	"testing"
6)
7
8const largeString = `Sed ut perspiciatis unde omnis iste natus error sit voluptatem accusantium doloremque laudantium, totam rem aperiam, eaque ipsa quae ab illo inventore veritatis et quasi architecto beatae vitae dicta sunt explicabo. Nemo enim ipsam voluptatem quia voluptas sit aspernatur aut odit aut fugit, sed quia consequuntur magni dolores eos qui ratione voluptatem sequi nesciunt. Neque porro quisquam est, qui dolorem ipsum quia dolor sit amet, consectetur, adipisci velit, sed quia non numquam eius modi tempora incidunt ut labore et dolore magnam aliquam quaerat voluptatem. Ut enim ad minima veniam, quis nostrum exercitationem ullam corporis suscipit laboriosam, nisi ut aliquid ex ea commodi consequatur? Quis autem vel eum iure reprehenderit qui in ea voluptate velit esse quam nihil molestiae consequatur, vel illum qui dolorem eum fugiat quo voluptas nulla pariatur? At vero eos et accusamus et iusto odio dignissimos ducimus qui blanditiis praesentium voluptatum deleniti atque corrupti quos dolores et quas molestias except`
9
10var snappyTestCases = map[string][]byte{
11	"REPEATREPEATREPEATREPEATREPEATREPEAT": {36, 20, 82, 69, 80, 69, 65, 84, 118, 6, 0},
12	"REALLY SHORT":                         {12, 44, 82, 69, 65, 76, 76, 89, 32, 83, 72, 79, 82, 84},
13	"AXBXCXDXEXFX":                         {12, 44, 65, 88, 66, 88, 67, 88, 68, 88, 69, 88, 70, 88},
14}
15
16var snappyStreamTestCases = map[string][]byte{
17	"PLAINDATA":                         {130, 83, 78, 65, 80, 80, 89, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 11, 9, 32, 80, 76, 65, 73, 78, 68, 65, 84, 65},
18	`{"a":"UtaitILHMDAAAAfU","b":"日本"}`: {130, 83, 78, 65, 80, 80, 89, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 39, 37, 144, 123, 34, 97, 34, 58, 34, 85, 116, 97, 105, 116, 73, 76, 72, 77, 68, 65, 65, 65, 65, 102, 85, 34, 44, 34, 98, 34, 58, 34, 230, 151, 165, 230, 156, 172, 34, 125},
19	largeString: {130, 83, 78, 65, 80, 80, 89, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 3, 89, 128, 8, 240, 90, 83, 101, 100, 32, 117, 116, 32, 112, 101, 114, 115, 112, 105, 99, 105, 97, 116, 105, 115, 32, 117, 110, 100, 101, 32, 111, 109, 110, 105, 115, 32, 105, 115, 116, 101, 32, 110, 97, 116, 117, 115, 32, 101, 114, 114, 111, 114, 32, 115, 105, 116, 32, 118, 111, 108, 117, 112, 116, 97, 116, 101, 109, 32, 97, 99, 99, 117, 115, 97, 110, 116, 105, 117, 109, 32, 100, 111, 108, 111, 114, 101, 109, 113, 117, 101, 32, 108, 97, 117, 100, 97, 5, 22, 240, 60, 44, 32, 116, 111, 116, 97, 109, 32, 114, 101, 109, 32, 97, 112, 101, 114, 105, 97, 109, 44, 32, 101, 97, 113, 117, 101, 32, 105, 112, 115, 97, 32, 113, 117, 97, 101, 32, 97, 98, 32, 105, 108, 108, 111, 32, 105, 110, 118, 101, 110, 116, 111, 114, 101, 32, 118, 101, 114, 105, 116, 97, 1, 141, 4, 101, 116, 1, 36, 88, 115, 105, 32, 97, 114, 99, 104, 105, 116, 101, 99, 116, 111, 32, 98, 101, 97, 116, 97, 101, 32, 118, 105, 1, 6, 120, 100, 105, 99, 116, 97, 32, 115, 117, 110, 116, 32, 101, 120, 112, 108, 105, 99, 97, 98, 111, 46, 32, 78, 101, 109, 111, 32, 101, 110, 105, 109, 5, 103, 0, 109, 46, 180, 0, 12, 113, 117, 105, 97, 17, 16, 0, 115, 5, 209, 72, 97, 115, 112, 101, 114, 110, 97, 116, 117, 114, 32, 97, 117, 116, 32, 111, 100, 105, 116, 5, 9, 36, 102, 117, 103, 105, 116, 44, 32, 115, 101, 100, 9, 53, 32, 99, 111, 110, 115, 101, 113, 117, 117, 110, 1, 42, 20, 109, 97, 103, 110, 105, 32, 9, 245, 16, 115, 32, 101, 111, 115, 1, 36, 28, 32, 114, 97, 116, 105, 111, 110, 101, 17, 96, 33, 36, 1, 51, 36, 105, 32, 110, 101, 115, 99, 105, 117, 110, 116, 1, 155, 1, 254, 16, 112, 111, 114, 114, 111, 1, 51, 36, 115, 113, 117, 97, 109, 32, 101, 115, 116, 44, 1, 14, 13, 81, 5, 183, 4, 117, 109, 1, 18, 0, 97, 9, 19, 4, 32, 115, 1, 149, 12, 109, 101, 116, 44, 9, 135, 76, 99, 116, 101, 116, 117, 114, 44, 32, 97, 100, 105, 112, 105, 115, 99, 105, 32, 118, 101, 108, 50, 173, 0, 24, 110, 111, 110, 32, 110, 117, 109, 9, 94, 84, 105, 117, 115, 32, 109, 111, 100, 105, 32, 116, 101, 109, 112, 111, 114, 97, 32, 105, 110, 99, 105, 100, 33, 52, 20, 117, 116, 32, 108, 97, 98, 33, 116, 4, 101, 116, 9, 106, 0, 101, 5, 219, 20, 97, 109, 32, 97, 108, 105, 5, 62, 33, 164, 8, 114, 97, 116, 29, 212, 12, 46, 32, 85, 116, 41, 94, 52, 97, 100, 32, 109, 105, 110, 105, 109, 97, 32, 118, 101, 110, 105, 33, 221, 72, 113, 117, 105, 115, 32, 110, 111, 115, 116, 114, 117, 109, 32, 101, 120, 101, 114, 99, 105, 33, 202, 104, 111, 110, 101, 109, 32, 117, 108, 108, 97, 109, 32, 99, 111, 114, 112, 111, 114, 105, 115, 32, 115, 117, 115, 99, 105, 112, 105, 13, 130, 8, 105, 111, 115, 1, 64, 12, 110, 105, 115, 105, 1, 150, 5, 126, 44, 105, 100, 32, 101, 120, 32, 101, 97, 32, 99, 111, 109, 5, 192, 0, 99, 41, 131, 33, 172, 8, 63, 32, 81, 1, 107, 4, 97, 117, 33, 101, 96, 118, 101, 108, 32, 101, 117, 109, 32, 105, 117, 114, 101, 32, 114, 101, 112, 114, 101, 104, 101, 110, 100, 101, 114, 105, 65, 63, 12, 105, 32, 105, 110, 1, 69, 16, 118, 111, 108, 117, 112, 65, 185, 1, 47, 24, 105, 116, 32, 101, 115, 115, 101, 1, 222, 64, 109, 32, 110, 105, 104, 105, 108, 32, 109, 111, 108, 101, 115, 116, 105, 97, 101, 46, 103, 0, 0, 44, 1, 45, 16, 32, 105, 108, 108, 117, 37, 143, 45, 36, 0, 109, 5, 110, 65, 33, 20, 97, 116, 32, 113, 117, 111, 17, 92, 44, 115, 32, 110, 117, 108, 108, 97, 32, 112, 97, 114, 105, 9, 165, 24, 65, 116, 32, 118, 101, 114, 111, 69, 34, 44, 101, 116, 32, 97, 99, 99, 117, 115, 97, 109, 117, 115, 1, 13, 104, 105, 117, 115, 116, 111, 32, 111, 100, 105, 111, 32, 100, 105, 103, 110, 105, 115, 115, 105, 109, 111, 115, 32, 100, 117, 99, 105, 1, 34, 80, 113, 117, 105, 32, 98, 108, 97, 110, 100, 105, 116, 105, 105, 115, 32, 112, 114, 97, 101, 115, 101, 101, 87, 17, 111, 56, 116, 117, 109, 32, 100, 101, 108, 101, 110, 105, 116, 105, 32, 97, 116, 65, 89, 28, 99, 111, 114, 114, 117, 112, 116, 105, 1, 150, 0, 115, 13, 174, 5, 109, 8, 113, 117, 97, 65, 5, 52, 108, 101, 115, 116, 105, 97, 115, 32, 101, 120, 99, 101, 112, 116, 0, 0, 0, 1, 0},
20}
21
22func makeMassive(input string, numCopies int) string {
23	outBuff := make([]byte, len(input) * numCopies)
24
25    for i := 0; i < numCopies; i++ {
26        copy(outBuff[len(outBuff):], input)
27    }
28
29    return string(outBuff)
30}
31
32func TestSnappyEncode(t *testing.T) {
33	for src, exp := range snappyTestCases {
34		dst := Encode([]byte(src))
35		if !bytes.Equal(dst, exp) {
36			t.Errorf("Expected %s to generate %v, but was %v", src, exp, dst)
37		}
38	}
39}
40
41func TestSnappyEncodeStream(t *testing.T) {
42	for src, _ := range snappyStreamTestCases {
43		dst := EncodeStream(nil, []byte(src))
44
45		// Block size can change the bytes generated, so let's just decode and make sure in matches out
46		dec, err := Decode(dst)
47		if err != nil {
48			t.Error(err)
49		}
50		if src != string(dec) {
51			t.Errorf("Expected decode to match encode orig = %s, decoded = %s", src, string(dec))
52		}
53	}
54}
55
56func TestSnappyLargeStringEncodeStream(t *testing.T) {
57	massiveString := makeMassive(largeString, 10000)
58	dst := EncodeStream(nil, []byte(massiveString))
59	dec, err := Decode(dst)
60	if err != nil {
61		t.Error(err)
62	}
63	if massiveString != string(dec) {
64		t.Errorf("Decoded string didn't match original input (not printing due to size)")
65	}
66}
67
68func TestSnappyDecode(t *testing.T) {
69	for exp, src := range snappyTestCases {
70		dst, err := Decode(src)
71		if err != nil {
72			t.Error("Encoding error: ", err)
73		} else if !bytes.Equal(dst, []byte(exp)) {
74			t.Errorf("Expected %s to be generated from %v, but was %s", exp, src, string(dst))
75		}
76	}
77}
78
79func TestSnappyDecodeStreams(t *testing.T) {
80	for exp, src := range snappyStreamTestCases {
81		dst, err := Decode(src)
82		if err != nil {
83			t.Error("Encoding error: ", err)
84		} else if !bytes.Equal(dst, []byte(exp)) {
85			t.Errorf("Expected %s to be generated from [%d]byte, but was %s", exp, len(src), string(dst))
86		}
87	}
88}
89
90func TestSnappyDecodeMalformedTruncatedHeader(t *testing.T) {
91	// Truncated headers should not cause a panic.
92	for i := 0; i < len(xerialHeader); i++ {
93		buf := make([]byte, i)
94		copy(buf, xerialHeader[:i])
95		if _, err := Decode(buf); err != ErrMalformed {
96			t.Errorf("expected ErrMalformed got %v", err)
97		}
98	}
99}
100
101func TestSnappyDecodeMalformedTruncatedSize(t *testing.T) {
102	// Inputs with valid Xerial header but truncated "size" field
103	sizes := []int{sizeOffset + 1, sizeOffset + 2, sizeOffset + 3}
104	for _, size := range sizes {
105		buf := make([]byte, size)
106		copy(buf, xerialHeader)
107		if _, err := Decode(buf); err != ErrMalformed {
108			t.Errorf("expected ErrMalformed got %v", err)
109		}
110	}
111}
112
113func TestSnappyDecodeMalformedBNoData(t *testing.T) {
114	// No data after the size field
115	buf := make([]byte, 20)
116	copy(buf, xerialHeader)
117	// indicate that there's one byte of data to be read
118	buf[len(buf)-1] = 1
119	if _, err := Decode(buf); err != ErrMalformed {
120		t.Errorf("expected ErrMalformed got %v", err)
121	}
122}
123
124func TestSnappyMasterDecodeFailed(t *testing.T) {
125	buf := make([]byte, 21)
126	copy(buf, xerialHeader)
127	// indicate that there's one byte of data to be read
128	buf[len(buf)-2] = 1
129	// A payload which will not decode
130	buf[len(buf)-1] = 1
131	if _, err := Decode(buf); err == ErrMalformed || err == nil {
132		t.Errorf("unexpected err: %v", err)
133	}
134}
135
136func BenchmarkSnappyDecode(b *testing.B) {
137	b.ReportAllocs()
138	b.ResetTimer()
139
140	for n := 0; n < b.N; n++ {
141		bytes := 0
142		for _, test := range snappyTestCases {
143			dst, err := Decode(test)
144			if err != nil {
145				b.Error("Decoding error: ", err)
146			}
147			bytes += len(dst)
148		}
149		b.SetBytes(int64(bytes))
150	}
151}
152
153func BenchmarkSnappyDecodeInto(b *testing.B) {
154	b.ReportAllocs()
155	b.ResetTimer()
156
157	var (
158		dst []byte
159		err error
160	)
161
162	for n := 0; n < b.N; n++ {
163		bytes := 0
164		for _, test := range snappyTestCases {
165
166			dst, err = DecodeInto(dst, test)
167			if err != nil {
168				b.Error("Decoding error: ", err)
169			}
170			bytes += len(dst)
171		}
172		b.SetBytes(int64(bytes))
173	}
174}
175
176func BenchmarkSnappyStreamDecode(b *testing.B) {
177	b.ReportAllocs()
178	b.ResetTimer()
179
180	for n := 0; n < b.N; n++ {
181		bytes := 0
182		for _, test := range snappyStreamTestCases {
183			dst, err := Decode(test)
184			if err != nil {
185				b.Error("Decoding error: ", err)
186			}
187			bytes += len(dst)
188		}
189		b.SetBytes(int64(bytes))
190	}
191}
192
193func BenchmarkSnappyStreamDecodeInto(b *testing.B) {
194	b.ReportAllocs()
195	b.ResetTimer()
196
197	var (
198		dst = make([]byte, 1024, 1024)
199		err error
200	)
201
202	for n := 0; n < b.N; n++ {
203		bytes := 0
204		for _, test := range snappyStreamTestCases {
205			dst, err = DecodeInto(dst, test)
206			if err != nil {
207				b.Error("Decoding error: ", err)
208			}
209			bytes += len(dst)
210		}
211		b.SetBytes(int64(bytes))
212	}
213}
214func BenchmarkSnappyStreamDecodeMassive(b *testing.B) {
215	massiveString := makeMassive(largeString, 10000)
216	enc := EncodeStream(nil, []byte(massiveString))
217
218	b.ReportAllocs()
219	b.ResetTimer()
220	b.SetBytes(int64(len(massiveString)))
221
222	for n := 0; n < b.N; n++ {
223		_, err := Decode(enc)
224		if err != nil {
225			b.Error("Decoding error: ", err)
226		}
227	}
228}
229
230func BenchmarkSnappyStreamDecodeIntoMassive(b *testing.B) {
231	massiveString := makeMassive(largeString, 10000)
232	enc := EncodeStream(nil, []byte(massiveString))
233
234	b.ReportAllocs()
235	b.ResetTimer()
236	b.SetBytes(int64(len(massiveString)))
237
238	var (
239		dst = make([]byte, 1024, 1024)
240		err error
241	)
242
243	for n := 0; n < b.N; n++ {
244		dst, err = DecodeInto(dst, enc)
245		if err != nil {
246			b.Error("Decoding error: ", err)
247		}
248	}
249}
250