mirror of
https://github.com/fankes/beszel.git
synced 2025-10-18 17:29:28 +08:00
141 lines
2.9 KiB
Go
141 lines
2.9 KiB
Go
package ghupdate
|
|
|
|
import (
|
|
"archive/tar"
|
|
"archive/zip"
|
|
"compress/gzip"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
)
|
|
|
|
// extract extracts an archive file to the destination directory.
|
|
// Supports .zip and .tar.gz files based on the file extension.
|
|
func extract(srcPath, destDir string) error {
|
|
if strings.HasSuffix(srcPath, ".tar.gz") {
|
|
return extractTarGz(srcPath, destDir)
|
|
}
|
|
// Default to zip extraction
|
|
return extractZip(srcPath, destDir)
|
|
}
|
|
|
|
// extractTarGz extracts a tar.gz archive to the destination directory.
|
|
func extractTarGz(srcPath, destDir string) error {
|
|
src, err := os.Open(srcPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer src.Close()
|
|
|
|
gz, err := gzip.NewReader(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer gz.Close()
|
|
|
|
tr := tar.NewReader(gz)
|
|
|
|
for {
|
|
header, err := tr.Next()
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if header.Typeflag == tar.TypeDir {
|
|
if err := os.MkdirAll(filepath.Join(destDir, header.Name), 0755); err != nil {
|
|
return err
|
|
}
|
|
continue
|
|
}
|
|
|
|
if err := os.MkdirAll(filepath.Dir(filepath.Join(destDir, header.Name)), 0755); err != nil {
|
|
return err
|
|
}
|
|
|
|
outFile, err := os.Create(filepath.Join(destDir, header.Name))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, err := io.Copy(outFile, tr); err != nil {
|
|
outFile.Close()
|
|
return err
|
|
}
|
|
outFile.Close()
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// extractZip extracts the zip archive at "src" to "dest".
|
|
//
|
|
// Note that only dirs and regular files will be extracted.
|
|
// Symbolic links, named pipes, sockets, or any other irregular files
|
|
// are skipped because they come with too many edge cases and ambiguities.
|
|
func extractZip(src, dest string) error {
|
|
zr, err := zip.OpenReader(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer zr.Close()
|
|
|
|
// normalize dest path to check later for Zip Slip
|
|
dest = filepath.Clean(dest) + string(os.PathSeparator)
|
|
|
|
for _, f := range zr.File {
|
|
err := extractFile(f, dest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// extractFile extracts the provided zipFile into "basePath/zipFileName" path,
|
|
// creating all the necessary path directories.
|
|
func extractFile(zipFile *zip.File, basePath string) error {
|
|
path := filepath.Join(basePath, zipFile.Name)
|
|
|
|
// check for Zip Slip
|
|
if !strings.HasPrefix(path, basePath) {
|
|
return fmt.Errorf("invalid file path: %s", path)
|
|
}
|
|
|
|
r, err := zipFile.Open()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer r.Close()
|
|
|
|
// allow only dirs or regular files
|
|
if zipFile.FileInfo().IsDir() {
|
|
if err := os.MkdirAll(path, os.ModePerm); err != nil {
|
|
return err
|
|
}
|
|
} else if zipFile.FileInfo().Mode().IsRegular() {
|
|
// ensure that the file path directories are created
|
|
if err := os.MkdirAll(filepath.Dir(path), os.ModePerm); err != nil {
|
|
return err
|
|
}
|
|
|
|
f, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, zipFile.Mode())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer f.Close()
|
|
|
|
_, err = io.Copy(f, r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|