cli/pkg/cmd/run/download/zip.go
Mislav Marohnić f8c7fd1d28 Fix extracting workflow artifact to a relative path
To prevent zipslip, we verify that each extracted file would fall
strictly under the prefix of the path to extract to. However, this
yielded a false positive when extracting to `.`, which is the default
for downloading a single archive.
2021-04-13 19:15:14 +02:00

69 lines
1.2 KiB
Go

package download
import (
"archive/zip"
"fmt"
"io"
"os"
"path/filepath"
"strings"
)
const (
dirMode os.FileMode = 0755
fileMode os.FileMode = 0644
execMode os.FileMode = 0755
)
func extractZip(zr *zip.Reader, destDir string) error {
destDirAbs, err := filepath.Abs(destDir)
if err != nil {
return err
}
pathPrefix := destDirAbs + string(filepath.Separator)
for _, zf := range zr.File {
fpath := filepath.Join(destDirAbs, filepath.FromSlash(zf.Name))
if !strings.HasPrefix(fpath, pathPrefix) {
continue
}
if err := extractZipFile(zf, fpath); err != nil {
return fmt.Errorf("error extracting %q: %w", zf.Name, err)
}
}
return nil
}
func extractZipFile(zf *zip.File, dest string) error {
zm := zf.Mode()
if zm.IsDir() {
return os.MkdirAll(dest, dirMode)
}
f, err := zf.Open()
if err != nil {
return err
}
defer f.Close()
if dir := filepath.Dir(dest); dir != "." {
if err := os.MkdirAll(dir, dirMode); err != nil {
return err
}
}
df, err := os.OpenFile(dest, os.O_WRONLY|os.O_CREATE|os.O_EXCL, getPerm(zm))
if err != nil {
return err
}
defer df.Close()
_, err = io.Copy(df, f)
return err
}
func getPerm(m os.FileMode) os.FileMode {
if m&0111 == 0 {
return fileMode
}
return execMode
}