feat: Add ZStandard compression support

This commit is contained in:
Felix Pojtinger
2021-11-30 18:44:27 +01:00
parent 7cd97c4d2d
commit e3d8af7780
4 changed files with 99 additions and 12 deletions

View File

@@ -10,6 +10,7 @@ import (
"path/filepath"
"strconv"
"github.com/klauspost/compress/zstd"
"github.com/klauspost/pgzip"
"github.com/pierrec/lz4/v4"
"github.com/pojntfx/stfs/pkg/adapters"
@@ -253,6 +254,43 @@ func archive(
hdr.Size = int64(fileSizeCounter.BytesRead)
hdr.Name += compressionFormatLZ4Suffix
case compressionFormatZStandardKey:
// Get the compressed size for the header
file, err := os.Open(path)
if err != nil {
return err
}
fileSizeCounter := counters.CounterWriter{
Writer: io.Discard,
}
zz, err := zstd.NewWriter(&fileSizeCounter)
if err != nil {
return err
}
if _, err := io.Copy(zz, file); err != nil {
return err
}
if err := zz.Flush(); err != nil {
return err
}
if err := zz.Close(); err != nil {
return err
}
if err := file.Close(); err != nil {
return err
}
if hdr.PAXRecords == nil {
hdr.PAXRecords = map[string]string{}
}
hdr.PAXRecords[pax.STFSRecordUncompressedSize] = strconv.Itoa(int(hdr.Size))
hdr.Size = int64(fileSizeCounter.BytesRead)
hdr.Name += compressionFormatZStandardSuffix
case compressionFormatNoneKey:
default:
return errUnsupportedCompressionFormat
@@ -352,6 +390,42 @@ func archive(
if err := file.Close(); err != nil {
return err
}
case compressionFormatZStandardKey:
// Compress and write the file
file, err := os.Open(path)
if err != nil {
return err
}
zz, err := zstd.NewWriter(tw)
if err != nil {
return err
}
if _, err := io.Copy(zz, file); err != nil {
return err
}
if isRegular {
if _, err := io.Copy(zz, file); err != nil {
return err
}
} else {
buf := make([]byte, controllers.BlockSize*recordSize)
if _, err := io.CopyBuffer(zz, file, buf); err != nil {
return err
}
}
if err := zz.Flush(); err != nil {
return err
}
if err := zz.Close(); err != nil {
return err
}
if err := file.Close(); err != nil {
return err
}
case compressionFormatNoneKey:
// Write the file
file, err := os.Open(path)

View File

@@ -8,6 +8,7 @@ import (
"os"
"path/filepath"
"github.com/klauspost/compress/zstd"
"github.com/klauspost/pgzip"
"github.com/pierrec/lz4/v4"
"github.com/pojntfx/stfs/pkg/controllers"
@@ -160,6 +161,15 @@ func restoreFromRecordAndBlock(
if _, err := io.Copy(dstFile, lz); err != nil {
return err
}
case compressionFormatZStandardKey:
zz, err := zstd.NewReader(tr)
if err != nil {
return err
}
if _, err := io.Copy(dstFile, zz); err != nil {
return err
}
case compressionFormatNoneKey:
if _, err := io.Copy(dstFile, tr); err != nil {
return err

View File

@@ -267,20 +267,20 @@ func indexHeader(
hdr.Size = int64(size)
}
switch compressionFormat {
case compressionFormatGZipKey:
fallthrough
case compressionFormatParallelGZipKey:
if hdr.FileInfo().Mode().IsRegular() {
if hdr.FileInfo().Mode().IsRegular() {
switch compressionFormat {
case compressionFormatGZipKey:
fallthrough
case compressionFormatParallelGZipKey:
hdr.Name = strings.TrimSuffix(hdr.Name, compressionFormatGZipSuffix)
}
case compressionFormatLZ4Key:
if hdr.FileInfo().Mode().IsRegular() {
case compressionFormatLZ4Key:
hdr.Name = strings.TrimSuffix(hdr.Name, compressionFormatLZ4Suffix)
case compressionFormatZStandardKey:
hdr.Name = strings.TrimSuffix(hdr.Name, compressionFormatZStandardSuffix)
case compressionFormatNoneKey:
default:
return errUnsupportedCompressionFormat
}
case compressionFormatNoneKey:
default:
return errUnsupportedCompressionFormat
}
if err := formatting.PrintCSV(formatting.GetTARHeaderAsCSV(record, block, hdr)); err != nil {

View File

@@ -26,10 +26,13 @@ const (
compressionFormatLZ4Key = "lz4"
compressionFormatLZ4Suffix = ".lz4"
compressionFormatZStandardKey = "zstandard"
compressionFormatZStandardSuffix = ".zst"
)
var (
knownCompressionFormats = []string{compressionFormatNoneKey, compressionFormatGZipKey, compressionFormatParallelGZipKey, compressionFormatLZ4Key}
knownCompressionFormats = []string{compressionFormatNoneKey, compressionFormatGZipKey, compressionFormatParallelGZipKey, compressionFormatLZ4Key, compressionFormatZStandardKey}
errUnknownCompressionFormat = errors.New("unknown compression format")
errUnsupportedCompressionFormat = errors.New("unsupported compression format")