mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2024-12-31 20:28:46 +00:00
684b7a999f
* Dump: Use mholt/archive/v3 to support tar including many compressions Signed-off-by: Philipp Homann <homann.philipp@googlemail.com> * Dump: Allow dump output to stdout Signed-off-by: Philipp Homann <homann.philipp@googlemail.com> * Dump: Fixed bug present since #6677 where SessionConfig.Provider is never "file" Signed-off-by: Philipp Homann <homann.philipp@googlemail.com> * Dump: never pack RepoRootPath, LFS.ContentPath and LogRootPath when they are below AppDataPath Signed-off-by: Philipp Homann <homann.philipp@googlemail.com> * Dump: also dump LFS (fixes #10058) Signed-off-by: Philipp Homann <homann.philipp@googlemail.com> * Dump: never dump CustomPath if CustomPath is a subdir of or equal to AppDataPath (fixes #10365) Signed-off-by: Philipp Homann <homann.philipp@googlemail.com> * Use log.Info instead of fmt.Fprintf Signed-off-by: Philipp Homann <homann.philipp@googlemail.com> * import ordering * make fmt Co-authored-by: zeripath <art27@cantab.net> Co-authored-by: techknowlogick <techknowlogick@gitea.io> Co-authored-by: Matti R <matti@mdranta.net>
369 lines
9.5 KiB
Go
369 lines
9.5 KiB
Go
package rardecode
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"errors"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"time"
|
|
)
|
|
|
|
// FileHeader HostOS types
|
|
const (
|
|
HostOSUnknown = 0
|
|
HostOSMSDOS = 1
|
|
HostOSOS2 = 2
|
|
HostOSWindows = 3
|
|
HostOSUnix = 4
|
|
HostOSMacOS = 5
|
|
HostOSBeOS = 6
|
|
)
|
|
|
|
const (
|
|
maxPassword = 128
|
|
)
|
|
|
|
var (
|
|
errShortFile = errors.New("rardecode: decoded file too short")
|
|
errInvalidFileBlock = errors.New("rardecode: invalid file block")
|
|
errUnexpectedArcEnd = errors.New("rardecode: unexpected end of archive")
|
|
errBadFileChecksum = errors.New("rardecode: bad file checksum")
|
|
)
|
|
|
|
type byteReader interface {
|
|
io.Reader
|
|
io.ByteReader
|
|
}
|
|
|
|
type limitedReader struct {
|
|
r io.Reader
|
|
n int64 // bytes remaining
|
|
shortErr error // error returned when r returns io.EOF with n > 0
|
|
}
|
|
|
|
func (l *limitedReader) Read(p []byte) (int, error) {
|
|
if l.n <= 0 {
|
|
return 0, io.EOF
|
|
}
|
|
if int64(len(p)) > l.n {
|
|
p = p[0:l.n]
|
|
}
|
|
n, err := l.r.Read(p)
|
|
l.n -= int64(n)
|
|
if err == io.EOF && l.n > 0 {
|
|
return n, l.shortErr
|
|
}
|
|
return n, err
|
|
}
|
|
|
|
type limitedByteReader struct {
|
|
limitedReader
|
|
br io.ByteReader
|
|
}
|
|
|
|
func (l *limitedByteReader) ReadByte() (byte, error) {
|
|
if l.n <= 0 {
|
|
return 0, io.EOF
|
|
}
|
|
c, err := l.br.ReadByte()
|
|
if err == nil {
|
|
l.n--
|
|
} else if err == io.EOF && l.n > 0 {
|
|
return 0, l.shortErr
|
|
}
|
|
return c, err
|
|
}
|
|
|
|
// limitByteReader returns a limitedByteReader that reads from r and stops with
|
|
// io.EOF after n bytes.
|
|
// If r returns an io.EOF before reading n bytes, io.ErrUnexpectedEOF is returned.
|
|
func limitByteReader(r byteReader, n int64) *limitedByteReader {
|
|
return &limitedByteReader{limitedReader{r, n, io.ErrUnexpectedEOF}, r}
|
|
}
|
|
|
|
// fileChecksum allows file checksum validations to be performed.
|
|
// File contents must first be written to fileChecksum. Then valid is
|
|
// called to perform the file checksum calculation to determine
|
|
// if the file contents are valid or not.
|
|
type fileChecksum interface {
|
|
io.Writer
|
|
valid() bool
|
|
}
|
|
|
|
// FileHeader represents a single file in a RAR archive.
|
|
type FileHeader struct {
|
|
Name string // file name using '/' as the directory separator
|
|
IsDir bool // is a directory
|
|
HostOS byte // Host OS the archive was created on
|
|
Attributes int64 // Host OS specific file attributes
|
|
PackedSize int64 // packed file size (or first block if the file spans volumes)
|
|
UnPackedSize int64 // unpacked file size
|
|
UnKnownSize bool // unpacked file size is not known
|
|
ModificationTime time.Time // modification time (non-zero if set)
|
|
CreationTime time.Time // creation time (non-zero if set)
|
|
AccessTime time.Time // access time (non-zero if set)
|
|
Version int // file version
|
|
}
|
|
|
|
// Mode returns an os.FileMode for the file, calculated from the Attributes field.
|
|
func (f *FileHeader) Mode() os.FileMode {
|
|
var m os.FileMode
|
|
|
|
if f.IsDir {
|
|
m = os.ModeDir
|
|
}
|
|
if f.HostOS == HostOSWindows {
|
|
if f.IsDir {
|
|
m |= 0777
|
|
} else if f.Attributes&1 > 0 {
|
|
m |= 0444 // readonly
|
|
} else {
|
|
m |= 0666
|
|
}
|
|
return m
|
|
}
|
|
// assume unix perms for all remaining os types
|
|
m |= os.FileMode(f.Attributes) & os.ModePerm
|
|
|
|
// only check other bits on unix host created archives
|
|
if f.HostOS != HostOSUnix {
|
|
return m
|
|
}
|
|
|
|
if f.Attributes&0x200 != 0 {
|
|
m |= os.ModeSticky
|
|
}
|
|
if f.Attributes&0x400 != 0 {
|
|
m |= os.ModeSetgid
|
|
}
|
|
if f.Attributes&0x800 != 0 {
|
|
m |= os.ModeSetuid
|
|
}
|
|
|
|
// Check for additional file types.
|
|
if f.Attributes&0xF000 == 0xA000 {
|
|
m |= os.ModeSymlink
|
|
}
|
|
return m
|
|
}
|
|
|
|
// fileBlockHeader represents a file block in a RAR archive.
|
|
// Files may comprise one or more file blocks.
|
|
// Solid files retain decode tables and dictionary from previous solid files in the archive.
|
|
type fileBlockHeader struct {
|
|
first bool // first block in file
|
|
last bool // last block in file
|
|
solid bool // file is solid
|
|
winSize uint // log base 2 of decode window size
|
|
cksum fileChecksum // file checksum
|
|
decoder decoder // decoder to use for file
|
|
key []byte // key for AES, non-empty if file encrypted
|
|
iv []byte // iv for AES, non-empty if file encrypted
|
|
FileHeader
|
|
}
|
|
|
|
// fileBlockReader provides sequential access to file blocks in a RAR archive.
|
|
type fileBlockReader interface {
|
|
io.Reader // Read's read data from the current file block
|
|
io.ByteReader // Read bytes from current file block
|
|
next() (*fileBlockHeader, error) // reads the next file block header at current position
|
|
reset() // resets encryption
|
|
isSolid() bool // is archive solid
|
|
version() int // returns current archive format version
|
|
}
|
|
|
|
// packedFileReader provides sequential access to packed files in a RAR archive.
|
|
type packedFileReader struct {
|
|
r fileBlockReader
|
|
h *fileBlockHeader // current file header
|
|
}
|
|
|
|
// nextBlockInFile reads the next file block in the current file at the current
|
|
// archive file position, or returns an error if there is a problem.
|
|
// It is invalid to call this when already at the last block in the current file.
|
|
func (f *packedFileReader) nextBlockInFile() error {
|
|
h, err := f.r.next()
|
|
if err != nil {
|
|
if err == io.EOF {
|
|
// archive ended, but file hasn't
|
|
return errUnexpectedArcEnd
|
|
}
|
|
return err
|
|
}
|
|
if h.first || h.Name != f.h.Name {
|
|
return errInvalidFileBlock
|
|
}
|
|
f.h = h
|
|
return nil
|
|
}
|
|
|
|
// next advances to the next packed file in the RAR archive.
|
|
func (f *packedFileReader) next() (*fileBlockHeader, error) {
|
|
if f.h != nil {
|
|
// skip to last block in current file
|
|
for !f.h.last {
|
|
// discard remaining block data
|
|
if _, err := io.Copy(ioutil.Discard, f.r); err != nil {
|
|
return nil, err
|
|
}
|
|
if err := f.nextBlockInFile(); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
// discard last block data
|
|
if _, err := io.Copy(ioutil.Discard, f.r); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
var err error
|
|
f.h, err = f.r.next() // get next file block
|
|
if err != nil {
|
|
if err == errArchiveEnd {
|
|
return nil, io.EOF
|
|
}
|
|
return nil, err
|
|
}
|
|
if !f.h.first {
|
|
return nil, errInvalidFileBlock
|
|
}
|
|
return f.h, nil
|
|
}
|
|
|
|
// Read reads the packed data for the current file into p.
|
|
func (f *packedFileReader) Read(p []byte) (int, error) {
|
|
n, err := f.r.Read(p) // read current block data
|
|
for err == io.EOF { // current block empty
|
|
if n > 0 {
|
|
return n, nil
|
|
}
|
|
if f.h == nil || f.h.last {
|
|
return 0, io.EOF // last block so end of file
|
|
}
|
|
if err := f.nextBlockInFile(); err != nil {
|
|
return 0, err
|
|
}
|
|
n, err = f.r.Read(p) // read new block data
|
|
}
|
|
return n, err
|
|
}
|
|
|
|
func (f *packedFileReader) ReadByte() (byte, error) {
|
|
c, err := f.r.ReadByte() // read current block data
|
|
for err == io.EOF && f.h != nil && !f.h.last { // current block empty
|
|
if err := f.nextBlockInFile(); err != nil {
|
|
return 0, err
|
|
}
|
|
c, err = f.r.ReadByte() // read new block data
|
|
}
|
|
return c, err
|
|
}
|
|
|
|
// Reader provides sequential access to files in a RAR archive.
|
|
type Reader struct {
|
|
r io.Reader // reader for current unpacked file
|
|
pr packedFileReader // reader for current packed file
|
|
dr decodeReader // reader for decoding and filters if file is compressed
|
|
cksum fileChecksum // current file checksum
|
|
solidr io.Reader // reader for solid file
|
|
}
|
|
|
|
// Read reads from the current file in the RAR archive.
|
|
func (r *Reader) Read(p []byte) (int, error) {
|
|
n, err := r.r.Read(p)
|
|
if err == io.EOF && r.cksum != nil && !r.cksum.valid() {
|
|
return n, errBadFileChecksum
|
|
}
|
|
return n, err
|
|
}
|
|
|
|
// Next advances to the next file in the archive.
|
|
func (r *Reader) Next() (*FileHeader, error) {
|
|
if r.solidr != nil {
|
|
// solid files must be read fully to update decoder information
|
|
if _, err := io.Copy(ioutil.Discard, r.solidr); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
h, err := r.pr.next() // skip to next file
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
r.solidr = nil
|
|
|
|
br := byteReader(&r.pr) // start with packed file reader
|
|
|
|
// check for encryption
|
|
if len(h.key) > 0 && len(h.iv) > 0 {
|
|
br = newAesDecryptReader(br, h.key, h.iv) // decrypt
|
|
}
|
|
r.r = br
|
|
// check for compression
|
|
if h.decoder != nil {
|
|
err = r.dr.init(br, h.decoder, h.winSize, !h.solid)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
r.r = &r.dr
|
|
if r.pr.r.isSolid() {
|
|
r.solidr = r.r
|
|
}
|
|
}
|
|
if h.UnPackedSize >= 0 && !h.UnKnownSize {
|
|
// Limit reading to UnPackedSize as there may be padding
|
|
r.r = &limitedReader{r.r, h.UnPackedSize, errShortFile}
|
|
}
|
|
r.cksum = h.cksum
|
|
if r.cksum != nil {
|
|
r.r = io.TeeReader(r.r, h.cksum) // write file data to checksum as it is read
|
|
}
|
|
fh := new(FileHeader)
|
|
*fh = h.FileHeader
|
|
return fh, nil
|
|
}
|
|
|
|
func (r *Reader) init(fbr fileBlockReader) {
|
|
r.r = bytes.NewReader(nil) // initial reads will always return EOF
|
|
r.pr.r = fbr
|
|
}
|
|
|
|
// NewReader creates a Reader reading from r.
|
|
// NewReader only supports single volume archives.
|
|
// Multi-volume archives must use OpenReader.
|
|
func NewReader(r io.Reader, password string) (*Reader, error) {
|
|
br, ok := r.(*bufio.Reader)
|
|
if !ok {
|
|
br = bufio.NewReader(r)
|
|
}
|
|
fbr, err := newFileBlockReader(br, password)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
rr := new(Reader)
|
|
rr.init(fbr)
|
|
return rr, nil
|
|
}
|
|
|
|
type ReadCloser struct {
|
|
v *volume
|
|
Reader
|
|
}
|
|
|
|
// Close closes the rar file.
|
|
func (rc *ReadCloser) Close() error {
|
|
return rc.v.Close()
|
|
}
|
|
|
|
// OpenReader opens a RAR archive specified by the name and returns a ReadCloser.
|
|
func OpenReader(name, password string) (*ReadCloser, error) {
|
|
v, err := openVolume(name, password)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
rc := new(ReadCloser)
|
|
rc.v = v
|
|
rc.Reader.init(v)
|
|
return rc, nil
|
|
}
|