Add zip2zip tool to copy zip entries from one file to another
This doesn't do any decompression / recompression, but just copies over the already compressed contents. So it's similar to zip -U, but allows rewriting of the paths. The first expected usecase is to replace img_from_target_files during the build, since it does the equivalent of this: zip2zip -i <target-files.zip> -o <img.zip> OTA/android-info.txt:android-info.txt IMAGES/*:. Except it decompresses and recompresses the images, which takes over a minute instead of a few seconds. Change-Id: I88d0df188635088783223873f78e193272dbdf1c
This commit is contained in:
10
Android.bp
10
Android.bp
@@ -10,6 +10,8 @@
|
|||||||
// 2) Build again
|
// 2) Build again
|
||||||
//
|
//
|
||||||
|
|
||||||
|
subdirs = ["third_party/zip"]
|
||||||
|
|
||||||
bootstrap_go_binary {
|
bootstrap_go_binary {
|
||||||
name: "soong_build",
|
name: "soong_build",
|
||||||
deps: [
|
deps: [
|
||||||
@@ -194,6 +196,14 @@ bootstrap_go_package {
|
|||||||
pluginFor: ["soong_build"],
|
pluginFor: ["soong_build"],
|
||||||
}
|
}
|
||||||
|
|
||||||
|
blueprint_go_binary {
|
||||||
|
name: "zip2zip",
|
||||||
|
deps: ["android-archive-zip"],
|
||||||
|
srcs: [
|
||||||
|
"cmd/zip2zip/zip2zip.go",
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
blueprint_go_binary {
|
blueprint_go_binary {
|
||||||
name: "soong_jar",
|
name: "soong_jar",
|
||||||
srcs: [
|
srcs: [
|
||||||
|
128
cmd/zip2zip/zip2zip.go
Normal file
128
cmd/zip2zip/zip2zip.go
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
// Copyright 2016 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 (
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"android/soong/third_party/zip"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
input = flag.String("i", "", "zip file to read from")
|
||||||
|
output = flag.String("o", "", "output file")
|
||||||
|
)
|
||||||
|
|
||||||
|
func usage() {
|
||||||
|
fmt.Fprintln(os.Stderr, "usage: zip2zip -i zipfile -o zipfile [filespec]...")
|
||||||
|
flag.PrintDefaults()
|
||||||
|
fmt.Fprintln(os.Stderr, " filespec:")
|
||||||
|
fmt.Fprintln(os.Stderr, " <name>")
|
||||||
|
fmt.Fprintln(os.Stderr, " <in_name>:<out_name>")
|
||||||
|
fmt.Fprintln(os.Stderr, " <glob>:<out_dir>/")
|
||||||
|
fmt.Fprintln(os.Stderr, "")
|
||||||
|
fmt.Fprintln(os.Stderr, "Files will be copied with their existing compression from the input zipfile to")
|
||||||
|
fmt.Fprintln(os.Stderr, "the output zipfile, in the order of filespec arguments")
|
||||||
|
os.Exit(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
flag.Parse()
|
||||||
|
|
||||||
|
if flag.NArg() == 0 || *input == "" || *output == "" {
|
||||||
|
usage()
|
||||||
|
}
|
||||||
|
|
||||||
|
reader, err := zip.OpenReader(*input)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintln(os.Stderr, err)
|
||||||
|
os.Exit(3)
|
||||||
|
}
|
||||||
|
defer reader.Close()
|
||||||
|
|
||||||
|
output, err := os.Create(*output)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintln(os.Stderr, err)
|
||||||
|
os.Exit(4)
|
||||||
|
}
|
||||||
|
defer output.Close()
|
||||||
|
|
||||||
|
writer := zip.NewWriter(output)
|
||||||
|
defer func() {
|
||||||
|
err := writer.Close()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintln(os.Stderr, err)
|
||||||
|
os.Exit(5)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
for _, arg := range flag.Args() {
|
||||||
|
var input string
|
||||||
|
var output string
|
||||||
|
|
||||||
|
// Reserve escaping for future implementation, so make sure no
|
||||||
|
// one is using \ and expecting a certain behavior.
|
||||||
|
if strings.Contains(arg, "\\") {
|
||||||
|
fmt.Fprintln(os.Stderr, "\\ characters are not currently supported")
|
||||||
|
os.Exit(6)
|
||||||
|
}
|
||||||
|
|
||||||
|
args := strings.SplitN(arg, ":", 2)
|
||||||
|
input = args[0]
|
||||||
|
if len(args) == 2 {
|
||||||
|
output = args[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.IndexAny(input, "*?[") >= 0 {
|
||||||
|
for _, file := range reader.File {
|
||||||
|
if match, err := filepath.Match(input, file.Name); err != nil {
|
||||||
|
fmt.Fprintln(os.Stderr, err)
|
||||||
|
os.Exit(7)
|
||||||
|
} else if match {
|
||||||
|
var newFileName string
|
||||||
|
if output == "" {
|
||||||
|
newFileName = file.Name
|
||||||
|
} else {
|
||||||
|
_, name := filepath.Split(file.Name)
|
||||||
|
newFileName = filepath.Join(output, name)
|
||||||
|
}
|
||||||
|
err = writer.CopyFrom(file, newFileName)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintln(os.Stderr, err)
|
||||||
|
os.Exit(8)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if output == "" {
|
||||||
|
output = input
|
||||||
|
}
|
||||||
|
for _, file := range reader.File {
|
||||||
|
if input == file.Name {
|
||||||
|
err = writer.CopyFrom(file, output)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintln(os.Stderr, err)
|
||||||
|
os.Exit(8)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
31
third_party/zip/Android.bp
vendored
Normal file
31
third_party/zip/Android.bp
vendored
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
// Copyright 2016 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.
|
||||||
|
|
||||||
|
bootstrap_go_package {
|
||||||
|
name: "android-archive-zip",
|
||||||
|
pkgPath: "android/soong/third_party/zip",
|
||||||
|
srcs: [
|
||||||
|
"reader.go",
|
||||||
|
"register.go",
|
||||||
|
"struct.go",
|
||||||
|
"writer.go",
|
||||||
|
|
||||||
|
"android.go",
|
||||||
|
],
|
||||||
|
testSrcs: [
|
||||||
|
"reader_test.go",
|
||||||
|
"writer_test.go",
|
||||||
|
"zip_test.go",
|
||||||
|
],
|
||||||
|
}
|
70
third_party/zip/android.go
vendored
Normal file
70
third_party/zip/android.go
vendored
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
// Copyright 2016 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 zip
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (w *Writer) CopyFrom(orig *File, newName string) error {
|
||||||
|
if w.last != nil && !w.last.closed {
|
||||||
|
if err := w.last.close(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
w.last = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
fileHeader := orig.FileHeader
|
||||||
|
fileHeader.Name = newName
|
||||||
|
fh := &fileHeader
|
||||||
|
fh.Flags |= 0x8
|
||||||
|
|
||||||
|
h := &header{
|
||||||
|
FileHeader: fh,
|
||||||
|
offset: uint64(w.cw.count),
|
||||||
|
}
|
||||||
|
w.dir = append(w.dir, h)
|
||||||
|
|
||||||
|
if err := writeHeader(w.cw, fh); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy data
|
||||||
|
dataOffset, err := orig.DataOffset()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
io.Copy(w.cw, io.NewSectionReader(orig.zipr, dataOffset, int64(orig.CompressedSize64)))
|
||||||
|
|
||||||
|
// Write data descriptor.
|
||||||
|
var buf []byte
|
||||||
|
if fh.isZip64() {
|
||||||
|
buf = make([]byte, dataDescriptor64Len)
|
||||||
|
} else {
|
||||||
|
buf = make([]byte, dataDescriptorLen)
|
||||||
|
}
|
||||||
|
b := writeBuf(buf)
|
||||||
|
b.uint32(dataDescriptorSignature)
|
||||||
|
b.uint32(fh.CRC32)
|
||||||
|
if fh.isZip64() {
|
||||||
|
b.uint64(fh.CompressedSize64)
|
||||||
|
b.uint64(fh.UncompressedSize64)
|
||||||
|
} else {
|
||||||
|
b.uint32(fh.CompressedSize)
|
||||||
|
b.uint32(fh.UncompressedSize)
|
||||||
|
}
|
||||||
|
_, err = w.cw.Write(buf)
|
||||||
|
return err
|
||||||
|
}
|
Reference in New Issue
Block a user