zipindex
provides a size optimized representation of a zip file directory to allow
decompressing files inside a ZIP file without reading the file index every file.
It will only provide the minimal needed data for successful decompression and CRC checks.
Custom metadata can be stored per file and filtering can be performed on the incoming files.
Currently, up to 100 million files per zip file is supported. If a streaming format is added, this limit may be lifted.
Indexing is performed on the last part of a complete ZIP file.
Three methods can be used:
The zipindex.ReadDir
function allows parsing from a raw buffer from the end of the file.
If this isn't enough to read the directory zipindex.ErrNeedMoreData
is returned, which will
return how much data is needed to read the directory.
Alternatively, zipindex.ReadFile
will open a file on disk and read the directory from that.
Finally zipindex.ReaderAt
allows to read the index from anything supporting the
io.ReaderAt
interface.
By default, only "regular" files are indexed, meaning directories and other entries are skipped, as well as files for which a decompressor isn't registered.
A custom filter function can be provided to change the default filtering. This also allows adding custom data for each file if more information is needed.
See examples in the documentation
Before serializing it is recommended to run the OptimizeSize()
on the returned files.
This will sort the entries and remove any redundant CRC information.
The files are serialized using the Serialize()
method.
This will allow the information to be recreated using zipindex.DeserializeFiles
,
or to find a single file zipindex.FindSerialized
can be used.
See examples in the documentation
To read a file, you will need 1) the serialized index and once you have gotten the information for the file you want to decompress form the ZIP file, you will need to forward the zip file to the offset specified by the information returned from the index.
A file contains the following information:
type File struct {
Name string // Name of the file as stored in the zip.
CompressedSize64 uint64 // Size of compressed data, excluding ZIP headers.
UncompressedSize64 uint64 // Size of the Uncompressed data.
Offset int64 // Offset where file data header starts.
CRC32 uint32 // CRC of the uncompressed data.
Method uint16 // Storage method.
Flags uint16 // General purpose bit flag
Custom map[string]string
}
First an io.Reader
must be forwarded to the absolute offset in Offset
.
It is up to the caller to decide how to achieve that.
To open an individual file from the index use the (*File).Open(r io.Reader)
with the
forwarded Reader to open the content.
Similar to stdlib zip, not all methods/flags may be supported.
For expert users, (*File).OpenRaw
allows access to the compressed data.
By default, zipindex keeps files stored uncompressed or deflate compressed. This covers the most commonly seen compression methods.
Furthermore, files compressed with zstandard as method 93 will be preserved and can be read back.
Use RegisterDecompressor
to register non-standard decompressors.
zipindex is released under the Apache License v2.0. You can find the complete text in the file LICENSE.
zipindex contains code that is Copyright (c) 2009 The Go Authors. See GO_LICENSE
file for license.
Contributions are welcome, please send PRs for any enhancements.