Add tests for merge_zips

Test: merge_zip_test.go
Change-Id: I8fc577946b40cad193864aa2ebda1bef3a0e050e
This commit is contained in:
Colin Cross 2018-07-14 22:19:14 -07:00
parent 37f6d79c7e
commit 2486065c43
3 changed files with 279 additions and 6 deletions

View file

@ -21,5 +21,8 @@ blueprint_go_binary {
srcs: [ srcs: [
"merge_zips.go", "merge_zips.go",
], ],
testSrcs: [
"merge_zips_test.go",
],
} }

View file

@ -114,7 +114,7 @@ func main() {
log.Fatal(err) log.Fatal(err)
} }
defer reader.Close() defer reader.Close()
namedReader := namedZipReader{path: input, reader: reader} namedReader := namedZipReader{path: input, reader: &reader.Reader}
readers = append(readers, namedReader) readers = append(readers, namedReader)
} }
@ -132,7 +132,7 @@ func main() {
// do merge // do merge
err = mergeZips(readers, writer, *manifest, *entrypoint, *pyMain, *sortEntries, *emulateJar, *emulatePar, err = mergeZips(readers, writer, *manifest, *entrypoint, *pyMain, *sortEntries, *emulateJar, *emulatePar,
*stripDirEntries, *ignoreDuplicates) *stripDirEntries, *ignoreDuplicates, []string(stripFiles), []string(stripDirs), map[string]bool(zipsToNotStrip))
if err != nil { if err != nil {
log.Fatal(err) log.Fatal(err)
} }
@ -141,7 +141,7 @@ func main() {
// a namedZipReader reads a .zip file and can say which file it's reading // a namedZipReader reads a .zip file and can say which file it's reading
type namedZipReader struct { type namedZipReader struct {
path string path string
reader *zip.ReadCloser reader *zip.Reader
} }
// a zipEntryPath refers to a file contained in a zip // a zipEntryPath refers to a file contained in a zip
@ -224,7 +224,8 @@ type fileMapping struct {
} }
func mergeZips(readers []namedZipReader, writer *zip.Writer, manifest, entrypoint, pyMain string, func mergeZips(readers []namedZipReader, writer *zip.Writer, manifest, entrypoint, pyMain string,
sortEntries, emulateJar, emulatePar, stripDirEntries, ignoreDuplicates bool) error { sortEntries, emulateJar, emulatePar, stripDirEntries, ignoreDuplicates bool,
stripFiles, stripDirs []string, zipsToNotStrip map[string]bool) error {
sourceByDest := make(map[string]zipSource, 0) sourceByDest := make(map[string]zipSource, 0)
orderedMappings := []fileMapping{} orderedMappings := []fileMapping{}
@ -338,7 +339,7 @@ func mergeZips(readers []namedZipReader, writer *zip.Writer, manifest, entrypoin
for _, namedReader := range readers { for _, namedReader := range readers {
_, skipStripThisZip := zipsToNotStrip[namedReader.path] _, skipStripThisZip := zipsToNotStrip[namedReader.path]
for _, file := range namedReader.reader.File { for _, file := range namedReader.reader.File {
if !skipStripThisZip && shouldStripFile(emulateJar, file.Name) { if !skipStripThisZip && shouldStripFile(emulateJar, stripFiles, stripDirs, file.Name) {
continue continue
} }
@ -419,7 +420,7 @@ func pathBeforeLastSlash(path string) string {
return ret return ret
} }
func shouldStripFile(emulateJar bool, name string) bool { func shouldStripFile(emulateJar bool, stripFiles, stripDirs []string, name string) bool {
for _, dir := range stripDirs { for _, dir := range stripDirs {
if strings.HasPrefix(name, dir+"/") { if strings.HasPrefix(name, dir+"/") {
if emulateJar { if emulateJar {

View file

@ -0,0 +1,269 @@
// Copyright 2018 Google Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package main
import (
"bytes"
"fmt"
"os"
"strconv"
"strings"
"testing"
"android/soong/jar"
"android/soong/third_party/zip"
)
type testZipEntry struct {
name string
mode os.FileMode
data []byte
}
var (
A = testZipEntry{"A", 0755, []byte("foo")}
a = testZipEntry{"a", 0755, []byte("foo")}
a2 = testZipEntry{"a", 0755, []byte("FOO2")}
a3 = testZipEntry{"a", 0755, []byte("Foo3")}
bDir = testZipEntry{"b/", os.ModeDir | 0755, nil}
bbDir = testZipEntry{"b/b/", os.ModeDir | 0755, nil}
bbb = testZipEntry{"b/b/b", 0755, nil}
ba = testZipEntry{"b/a", 0755, []byte("foob")}
bc = testZipEntry{"b/c", 0755, []byte("bar")}
bd = testZipEntry{"b/d", 0700, []byte("baz")}
be = testZipEntry{"b/e", 0700, []byte("")}
metainfDir = testZipEntry{jar.MetaDir, os.ModeDir | 0755, nil}
manifestFile = testZipEntry{jar.ManifestFile, 0755, []byte("manifest")}
manifestFile2 = testZipEntry{jar.ManifestFile, 0755, []byte("manifest2")}
moduleInfoFile = testZipEntry{jar.ModuleInfoClass, 0755, []byte("module-info")}
)
func TestMergeZips(t *testing.T) {
testCases := []struct {
name string
in [][]testZipEntry
stripFiles []string
stripDirs []string
jar bool
sort bool
ignoreDuplicates bool
stripDirEntries bool
zipsToNotStrip map[string]bool
out []testZipEntry
err string
}{
{
name: "duplicates error",
in: [][]testZipEntry{
{a},
{a2},
{a3},
},
out: []testZipEntry{a},
err: "duplicate",
},
{
name: "duplicates take first",
in: [][]testZipEntry{
{a},
{a2},
{a3},
},
out: []testZipEntry{a},
ignoreDuplicates: true,
},
{
name: "sort",
in: [][]testZipEntry{
{be, bc, bDir, bbDir, bbb, A, metainfDir, manifestFile},
},
out: []testZipEntry{A, metainfDir, manifestFile, bDir, bbDir, bbb, bc, be},
sort: true,
},
{
name: "jar sort",
in: [][]testZipEntry{
{be, bc, bDir, A, metainfDir, manifestFile},
},
out: []testZipEntry{metainfDir, manifestFile, A, bDir, bc, be},
jar: true,
},
{
name: "jar merge",
in: [][]testZipEntry{
{metainfDir, manifestFile, bDir, be},
{metainfDir, manifestFile2, bDir, bc},
{metainfDir, manifestFile2, A},
},
out: []testZipEntry{metainfDir, manifestFile, A, bDir, bc, be},
jar: true,
},
{
name: "merge",
in: [][]testZipEntry{
{bDir, be},
{bDir, bc},
{A},
},
out: []testZipEntry{bDir, be, bc, A},
},
{
name: "strip dir entries",
in: [][]testZipEntry{
{a, bDir, bbDir, bbb, bc, bd, be},
},
out: []testZipEntry{a, bbb, bc, bd, be},
stripDirEntries: true,
},
{
name: "strip file name",
in: [][]testZipEntry{
{a, bDir, ba},
},
out: []testZipEntry{bDir},
stripFiles: []string{"a"},
},
{
name: "strip dirs",
in: [][]testZipEntry{
{a, bDir, bbDir, bbb, bc, bd, be},
},
out: []testZipEntry{a},
stripDirs: []string{"b"},
},
{
name: "zips to not strip",
in: [][]testZipEntry{
{a, bDir, bc},
{bDir, bd},
{bDir, be},
},
out: []testZipEntry{a, bDir, bd},
stripDirs: []string{"b"},
zipsToNotStrip: map[string]bool{
"in1": true,
},
},
}
for _, test := range testCases {
t.Run(test.name, func(t *testing.T) {
var readers []namedZipReader
for i, in := range test.in {
r := testZipEntriesToZipReader(in)
readers = append(readers, namedZipReader{
path: "in" + strconv.Itoa(i),
reader: r,
})
}
want := testZipEntriesToBuf(test.out)
out := &bytes.Buffer{}
writer := zip.NewWriter(out)
err := mergeZips(readers, writer, "", "", "",
test.sort, test.jar, false, test.stripDirEntries, test.ignoreDuplicates,
test.stripFiles, test.stripDirs, test.zipsToNotStrip)
closeErr := writer.Close()
if closeErr != nil {
t.Fatal(err)
}
if test.err != "" {
if err == nil {
t.Fatal("missing err, expected: ", test.err)
} else if !strings.Contains(strings.ToLower(err.Error()), strings.ToLower(test.err)) {
t.Fatal("incorrect err, want:", test.err, "got:", err)
}
return
}
if !bytes.Equal(want, out.Bytes()) {
t.Error("incorrect zip output")
t.Errorf("want:\n%s", dumpZip(want))
t.Errorf("got:\n%s", dumpZip(out.Bytes()))
}
})
}
}
func testZipEntriesToBuf(entries []testZipEntry) []byte {
b := &bytes.Buffer{}
zw := zip.NewWriter(b)
for _, e := range entries {
fh := zip.FileHeader{
Name: e.name,
}
fh.SetMode(e.mode)
w, err := zw.CreateHeader(&fh)
if err != nil {
panic(err)
}
_, err = w.Write(e.data)
if err != nil {
panic(err)
}
}
err := zw.Close()
if err != nil {
panic(err)
}
return b.Bytes()
}
func testZipEntriesToZipReader(entries []testZipEntry) *zip.Reader {
b := testZipEntriesToBuf(entries)
r := bytes.NewReader(b)
zr, err := zip.NewReader(r, int64(len(b)))
if err != nil {
panic(err)
}
return zr
}
func dumpZip(buf []byte) string {
r := bytes.NewReader(buf)
zr, err := zip.NewReader(r, int64(len(buf)))
if err != nil {
panic(err)
}
var ret string
for _, f := range zr.File {
ret += fmt.Sprintf("%v: %v %v %08x\n", f.Name, f.Mode(), f.UncompressedSize64, f.CRC32)
}
return ret
}