mirror of
				https://gitea.com/Lydanne/buildx.git
				synced 2025-11-04 10:03:42 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			122 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2019+ Klaus Post. All rights reserved.
 | 
						|
// License information can be found in the LICENSE file.
 | 
						|
 | 
						|
package zstd
 | 
						|
 | 
						|
import (
 | 
						|
	"errors"
 | 
						|
	"io"
 | 
						|
	"sync"
 | 
						|
)
 | 
						|
 | 
						|
// ZipMethodWinZip is the method for Zstandard compressed data inside Zip files for WinZip.
 | 
						|
// See https://www.winzip.com/win/en/comp_info.html
 | 
						|
const ZipMethodWinZip = 93
 | 
						|
 | 
						|
// ZipMethodPKWare is the original method number used by PKWARE to indicate Zstandard compression.
 | 
						|
// Deprecated: This has been deprecated by PKWARE, use ZipMethodWinZip instead for compression.
 | 
						|
// See https://pkware.cachefly.net/webdocs/APPNOTE/APPNOTE-6.3.9.TXT
 | 
						|
const ZipMethodPKWare = 20
 | 
						|
 | 
						|
var zipReaderPool sync.Pool
 | 
						|
 | 
						|
// newZipReader cannot be used since we would leak goroutines...
 | 
						|
func newZipReader(r io.Reader) io.ReadCloser {
 | 
						|
	dec, ok := zipReaderPool.Get().(*Decoder)
 | 
						|
	if ok {
 | 
						|
		dec.Reset(r)
 | 
						|
	} else {
 | 
						|
		d, err := NewReader(r, WithDecoderConcurrency(1), WithDecoderLowmem(true))
 | 
						|
		if err != nil {
 | 
						|
			panic(err)
 | 
						|
		}
 | 
						|
		dec = d
 | 
						|
	}
 | 
						|
	return &pooledZipReader{dec: dec}
 | 
						|
}
 | 
						|
 | 
						|
type pooledZipReader struct {
 | 
						|
	mu  sync.Mutex // guards Close and Read
 | 
						|
	dec *Decoder
 | 
						|
}
 | 
						|
 | 
						|
func (r *pooledZipReader) Read(p []byte) (n int, err error) {
 | 
						|
	r.mu.Lock()
 | 
						|
	defer r.mu.Unlock()
 | 
						|
	if r.dec == nil {
 | 
						|
		return 0, errors.New("Read after Close")
 | 
						|
	}
 | 
						|
	dec, err := r.dec.Read(p)
 | 
						|
 | 
						|
	return dec, err
 | 
						|
}
 | 
						|
 | 
						|
func (r *pooledZipReader) Close() error {
 | 
						|
	r.mu.Lock()
 | 
						|
	defer r.mu.Unlock()
 | 
						|
	var err error
 | 
						|
	if r.dec != nil {
 | 
						|
		err = r.dec.Reset(nil)
 | 
						|
		zipReaderPool.Put(r.dec)
 | 
						|
		r.dec = nil
 | 
						|
	}
 | 
						|
	return err
 | 
						|
}
 | 
						|
 | 
						|
type pooledZipWriter struct {
 | 
						|
	mu  sync.Mutex // guards Close and Read
 | 
						|
	enc *Encoder
 | 
						|
}
 | 
						|
 | 
						|
func (w *pooledZipWriter) Write(p []byte) (n int, err error) {
 | 
						|
	w.mu.Lock()
 | 
						|
	defer w.mu.Unlock()
 | 
						|
	if w.enc == nil {
 | 
						|
		return 0, errors.New("Write after Close")
 | 
						|
	}
 | 
						|
	return w.enc.Write(p)
 | 
						|
}
 | 
						|
 | 
						|
func (w *pooledZipWriter) Close() error {
 | 
						|
	w.mu.Lock()
 | 
						|
	defer w.mu.Unlock()
 | 
						|
	var err error
 | 
						|
	if w.enc != nil {
 | 
						|
		err = w.enc.Close()
 | 
						|
		zipReaderPool.Put(w.enc)
 | 
						|
		w.enc = nil
 | 
						|
	}
 | 
						|
	return err
 | 
						|
}
 | 
						|
 | 
						|
// ZipCompressor returns a compressor that can be registered with zip libraries.
 | 
						|
// The provided encoder options will be used on all encodes.
 | 
						|
func ZipCompressor(opts ...EOption) func(w io.Writer) (io.WriteCloser, error) {
 | 
						|
	var pool sync.Pool
 | 
						|
	return func(w io.Writer) (io.WriteCloser, error) {
 | 
						|
		enc, ok := pool.Get().(*Encoder)
 | 
						|
		if ok {
 | 
						|
			enc.Reset(w)
 | 
						|
		} else {
 | 
						|
			var err error
 | 
						|
			enc, err = NewWriter(w, opts...)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return &pooledZipWriter{enc: enc}, nil
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// ZipDecompressor returns a decompressor that can be registered with zip libraries.
 | 
						|
// See ZipCompressor for example.
 | 
						|
func ZipDecompressor() func(r io.Reader) io.ReadCloser {
 | 
						|
	return func(r io.Reader) io.ReadCloser {
 | 
						|
		d, err := NewReader(r, WithDecoderConcurrency(1), WithDecoderLowmem(true))
 | 
						|
		if err != nil {
 | 
						|
			panic(err)
 | 
						|
		}
 | 
						|
		return d.IOReadCloser()
 | 
						|
	}
 | 
						|
}
 |