Files
golib/archive/helper/interface.go
Salim Amine Bou Aram cd083bb53f Package archive/helper
- adding package to compress/uncompress with reader or writer
- refactor to allowing to use same source of io as result: io.reader or io.writer
- optimize code & buf to limit mem use
- rework variable to be thread safe

Package archive/compress
- add function DetectOnly to detect algo and return an updated reader but not the decompressor reader
- update function Detect to use DetectOnly to limit duplication code

Other
- bump dependencies
2024-11-27 17:23:14 +01:00

156 lines
3.6 KiB
Go

/*
* MIT License
*
* Copyright (c) 2024 Salim Amine BOU ARAM & Nicolas JUHEL
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*
*/
package helper
import (
"bytes"
"errors"
"io"
"sync"
"sync/atomic"
libarc "github.com/nabbar/golib/archive"
arccmp "github.com/nabbar/golib/archive/compress"
)
const chunkSize = 512
var (
ErrInvalidSource = errors.New("invalid source")
ErrClosedResource = errors.New("closed resource")
ErrInvalidOperation = errors.New("invalid operation")
)
type Helper interface {
io.ReadWriteCloser
}
func New(algo arccmp.Algorithm, ope Operation, src any) (h Helper, err error) {
if r, k := src.(io.Reader); k {
return NewReader(algo, ope, r)
}
if w, k := src.(io.Writer); k {
return NewWriter(algo, ope, w)
}
return nil, ErrInvalidSource
}
func NewReader(algo arccmp.Algorithm, ope Operation, src io.Reader) (Helper, error) {
switch ope {
case Compress:
return makeCompressReader(algo, src)
case Decompress:
return makeDeCompressReader(algo, src)
}
return nil, ErrInvalidOperation
}
func NewWriter(algo arccmp.Algorithm, ope Operation, dst io.Writer) (Helper, error) {
switch ope {
case Compress:
return makeCompressWriter(algo, dst)
case Decompress:
return makeDeCompressWriter(algo, dst)
}
return nil, ErrInvalidOperation
}
func makeCompressWriter(algo arccmp.Algorithm, src io.Writer) (h Helper, err error) {
wc, ok := src.(io.WriteCloser)
if !ok {
wc = libarc.NopWriteCloser(src)
}
if wc, err = algo.Writer(wc); err != nil {
return nil, err
} else {
return &compressWriter{
dst: wc,
}, nil
}
}
func makeCompressReader(algo arccmp.Algorithm, src io.Reader) (h Helper, err error) {
rc, ok := src.(io.ReadCloser)
if !ok {
rc = io.NopCloser(src)
}
var (
buf = bytes.NewBuffer(make([]byte, 0))
wrt io.WriteCloser
)
wrt, err = algo.Writer(libarc.NopWriteCloser(buf))
return &compressReader{
src: rc,
wrt: wrt,
buf: buf,
clo: new(atomic.Bool),
}, nil
}
func makeDeCompressReader(algo arccmp.Algorithm, src io.Reader) (h Helper, err error) {
rc, ok := src.(io.ReadCloser)
if !ok {
rc = io.NopCloser(src)
}
if rc, err = algo.Reader(rc); err != nil {
return nil, err
} else {
return &deCompressReader{
src: rc,
}, nil
}
}
func makeDeCompressWriter(algo arccmp.Algorithm, src io.Writer) (h Helper, err error) {
wc, ok := src.(io.WriteCloser)
if !ok {
wc = libarc.NopWriteCloser(src)
}
return &deCompressWriter{
alg: algo,
wrt: wc,
buf: &bufNoEOF{
m: sync.Mutex{},
b: bytes.NewBuffer(make([]byte, 0)),
c: new(atomic.Bool),
},
clo: new(atomic.Bool),
run: new(atomic.Bool),
}, nil
}