package ioutils import ( "bytes" "compress/gzip" "errors" "io" "os" "testing" "github.com/cometbft/cometbft/libs/rand" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) func TestUncompress(t *testing.T) { wasmRaw, err := os.ReadFile("../keeper/testdata/hackatom.wasm") require.NoError(t, err) wasmGzipped, err := os.ReadFile("../keeper/testdata/hackatom.wasm.gzip") require.NoError(t, err) const maxSize = 400_000 specs := map[string]struct { src []byte expError error expResult []byte }{ "handle wasm compressed": { src: wasmGzipped, expResult: wasmRaw, }, "handle gzip identifier only": { src: gzipIdent, expError: io.ErrUnexpectedEOF, }, "handle broken gzip": { src: append(gzipIdent, byte(0x1)), expError: io.ErrUnexpectedEOF, }, "handle incomplete gzip": { src: wasmGzipped[:len(wasmGzipped)-5], expError: io.ErrUnexpectedEOF, }, "handle limit gzip output": { src: asGzip(bytes.Repeat([]byte{0x1}, maxSize)), expResult: bytes.Repeat([]byte{0x1}, maxSize), }, "handle big gzip output": { src: asGzip(bytes.Repeat([]byte{0x1}, maxSize+1)), expError: errLimit, }, "handle big gzip archive": { src: asGzip(rand.Bytes(2 * maxSize)), expError: errLimit, }, } for msg, spec := range specs { t.Run(msg, func(t *testing.T) { r, err := Uncompress(spec.src, maxSize) require.True(t, errors.Is(err, spec.expError), "exp %v got %+v", spec.expError, err) if spec.expError != nil { return } assert.Equal(t, spec.expResult, r) }) } } func asGzip(src []byte) []byte { var buf bytes.Buffer zipper := gzip.NewWriter(&buf) if _, err := io.Copy(zipper, bytes.NewReader(src)); err != nil { panic(err) } if err := zipper.Close(); err != nil { panic(err) } return buf.Bytes() }