Skip to content

Commit

Permalink
chore: extracting compressed files to correct location (#1817)
Browse files Browse the repository at this point in the history
  • Loading branch information
ForestL18 authored Feb 4, 2025
1 parent 0ac6c3b commit e9b5feb
Showing 1 changed file with 213 additions and 9 deletions.
222 changes: 213 additions & 9 deletions component/updater/update_ui.go
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
package updater

import (
"archive/tar"
"archive/zip"
"compress/gzip"
"fmt"
"io"
"os"
Expand All @@ -22,6 +24,14 @@ type UIUpdater struct {
mutex sync.Mutex
}

type compressionType int

const (
typeUnknown compressionType = iota
typeZip
typeTarGzip
)

var DefaultUiUpdater = &UIUpdater{}

func NewUiUpdater(externalUI, externalUIURL, externalUIName string) *UIUpdater {
Expand Down Expand Up @@ -70,6 +80,24 @@ func (u *UIUpdater) DownloadUI() error {
return u.downloadUI()
}

func detectFileType(data []byte) compressionType {
if len(data) < 4 {
return typeUnknown
}

// Zip: 0x50 0x4B 0x03 0x04
if data[0] == 0x50 && data[1] == 0x4B && data[2] == 0x03 && data[3] == 0x04 {
return typeZip
}

// GZip: 0x1F 0x8B
if data[0] == 0x1F && data[1] == 0x8B {
return typeTarGzip
}

return typeUnknown
}

func (u *UIUpdater) downloadUI() error {
err := u.prepareUIPath()
if err != nil {
Expand All @@ -78,12 +106,23 @@ func (u *UIUpdater) downloadUI() error {

data, err := downloadForBytes(u.externalUIURL)
if err != nil {
return fmt.Errorf("can't download file: %w", err)
return fmt.Errorf("can't download file: %w", err)
}

saved := path.Join(C.Path.HomeDir(), "download.zip")
fileType := detectFileType(data)
if fileType == typeUnknown {
return fmt.Errorf("unknown or unsupported file type")
}

ext := ".zip"
if fileType == typeTarGzip {
ext = ".tgz"
}

saved := path.Join(C.Path.HomeDir(), "download"+ext)
log.Debugln("compression Type: %s", ext)
if err = saveFile(data, saved); err != nil {
return fmt.Errorf("can't save zip file: %w", err)
return fmt.Errorf("can't save compressed file: %w", err)
}
defer os.Remove(saved)

Expand All @@ -94,12 +133,12 @@ func (u *UIUpdater) downloadUI() error {
}
}

unzipFolder, err := unzip(saved, C.Path.HomeDir())
extractedFolder, err := extract(saved, C.Path.HomeDir())
if err != nil {
return fmt.Errorf("can't extract zip file: %w", err)
return fmt.Errorf("can't extract compressed file: %w", err)
}

err = os.Rename(unzipFolder, u.externalUIPath)
err = os.Rename(extractedFolder, u.externalUIPath)
if err != nil {
return fmt.Errorf("rename UI folder failed: %w", err)
}
Expand All @@ -122,9 +161,56 @@ func unzip(src, dest string) (string, error) {
return "", err
}
defer r.Close()

// check whether or not only exists singleRoot dir
rootDir := ""
isSingleRoot := true
for _, f := range r.File {
parts := strings.Split(strings.Trim(f.Name, "/"), "/")
if len(parts) == 0 {
continue
}

if len(parts) == 1 {
isSingleRoot = false
break
}

if rootDir == "" {
rootDir = parts[0]
} else if parts[0] != rootDir {
isSingleRoot = false
break
}
}

// build the dir of extraction
var extractedFolder string
if isSingleRoot && rootDir != "" {
// if the singleRoot, use it directly
extractedFolder = filepath.Join(dest, rootDir)
} else {
// or put the files/dirs into new dir
baseName := filepath.Base(src)
baseName = strings.TrimSuffix(baseName, filepath.Ext(baseName))
extractedFolder = filepath.Join(dest, baseName)

for i := 1; ; i++ {
if _, err := os.Stat(extractedFolder); os.IsNotExist(err) {
break
}
extractedFolder = filepath.Join(dest, fmt.Sprintf("%s_%d", baseName, i))
}
}

for _, f := range r.File {
fpath := filepath.Join(dest, f.Name)
var fpath string
if isSingleRoot && rootDir != "" {
fpath = filepath.Join(dest, f.Name)
} else {
fpath = filepath.Join(extractedFolder, f.Name)
}

if !strings.HasPrefix(fpath, filepath.Clean(dest)+string(os.PathSeparator)) {
return "", fmt.Errorf("invalid file path: %s", fpath)
}
Expand All @@ -149,13 +235,131 @@ func unzip(src, dest string) (string, error) {
if err != nil {
return "", err
}
if extractedFolder == "" {
extractedFolder = filepath.Dir(fpath)
}
return extractedFolder, nil
}

func untgz(src, dest string) (string, error) {
file, err := os.Open(src)
if err != nil {
return "", err
}
defer file.Close()

gzr, err := gzip.NewReader(file)
if err != nil {
return "", err
}
defer gzr.Close()

tr := tar.NewReader(gzr)

rootDir := ""
isSingleRoot := true
for {
header, err := tr.Next()
if err == io.EOF {
break
}
if err != nil {
return "", err
}

parts := strings.Split(strings.Trim(header.Name, "/"), "/")
if len(parts) == 0 {
continue
}

if len(parts) == 1 {
isSingleRoot = false
break
}

if rootDir == "" {
rootDir = parts[0]
} else if parts[0] != rootDir {
isSingleRoot = false
break
}
}

file.Seek(0, 0)
gzr, _ = gzip.NewReader(file)
tr = tar.NewReader(gzr)

var extractedFolder string
if isSingleRoot && rootDir != "" {
extractedFolder = filepath.Join(dest, rootDir)
} else {
baseName := filepath.Base(src)
baseName = strings.TrimSuffix(baseName, filepath.Ext(baseName))
baseName = strings.TrimSuffix(baseName, ".tar")
extractedFolder = filepath.Join(dest, baseName)

for i := 1; ; i++ {
if _, err := os.Stat(extractedFolder); os.IsNotExist(err) {
break
}
extractedFolder = filepath.Join(dest, fmt.Sprintf("%s_%d", baseName, i))
}
}

for {
header, err := tr.Next()
if err == io.EOF {
break
}
if err != nil {
return "", err
}

var fpath string
if isSingleRoot && rootDir != "" {
fpath = filepath.Join(dest, header.Name)
} else {
fpath = filepath.Join(extractedFolder, header.Name)
}

if !strings.HasPrefix(fpath, filepath.Clean(dest)+string(os.PathSeparator)) {
return "", fmt.Errorf("invalid file path: %s", fpath)
}

switch header.Typeflag {
case tar.TypeDir:
if err = os.MkdirAll(fpath, os.FileMode(header.Mode)); err != nil {
return "", err
}
case tar.TypeReg:
if err = os.MkdirAll(filepath.Dir(fpath), os.ModePerm); err != nil {
return "", err
}
outFile, err := os.OpenFile(fpath, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, os.FileMode(header.Mode))
if err != nil {
return "", err
}
if _, err := io.Copy(outFile, tr); err != nil {
outFile.Close()
return "", err
}
outFile.Close()
}
}
return extractedFolder, nil
}

func extract(src, dest string) (string, error) {
srcLower := strings.ToLower(src)
switch {
case strings.HasSuffix(srcLower, ".tar.gz") ||
strings.HasSuffix(srcLower, ".tgz"):
return untgz(src, dest)
case strings.HasSuffix(srcLower, ".zip"):
return unzip(src, dest)
default:
return "", fmt.Errorf("unsupported file format: %s", src)
}
}

func cleanup(root string) error {
if _, err := os.Stat(root); os.IsNotExist(err) {
return nil
Expand Down

0 comments on commit e9b5feb

Please sign in to comment.