mirror of https://github.com/portainer/portainer
				
				
				
			
		
			
				
	
	
		
			120 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			120 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Go
		
	
	
| package archive
 | |
| 
 | |
| import (
 | |
| 	"archive/tar"
 | |
| 	"compress/gzip"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"strings"
 | |
| )
 | |
| 
 | |
| // TarGzDir creates a tar.gz archive and returns it's path.
 | |
| // abosolutePath should be an absolute path to a directory.
 | |
| // Archive name will be <directoryName>.tar.gz and will be placed next to the directory.
 | |
| func TarGzDir(absolutePath string) (string, error) {
 | |
| 	targzPath := filepath.Join(absolutePath, fmt.Sprintf("%s.tar.gz", filepath.Base(absolutePath)))
 | |
| 	outFile, err := os.Create(targzPath)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	defer outFile.Close()
 | |
| 
 | |
| 	zipWriter := gzip.NewWriter(outFile)
 | |
| 	defer zipWriter.Close()
 | |
| 	tarWriter := tar.NewWriter(zipWriter)
 | |
| 	defer tarWriter.Close()
 | |
| 
 | |
| 	err = filepath.Walk(absolutePath, func(path string, info os.FileInfo, err error) error {
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		if path == targzPath {
 | |
| 			return nil // skip archive file
 | |
| 		}
 | |
| 
 | |
| 		pathInArchive := filepath.Clean(strings.TrimPrefix(path, absolutePath))
 | |
| 		if pathInArchive == "" {
 | |
| 			return nil // skip root dir
 | |
| 		}
 | |
| 
 | |
| 		return addToArchive(tarWriter, pathInArchive, path, info)
 | |
| 	})
 | |
| 
 | |
| 	return targzPath, err
 | |
| }
 | |
| 
 | |
| func addToArchive(tarWriter *tar.Writer, pathInArchive string, path string, info os.FileInfo) error {
 | |
| 	header, err := tar.FileInfoHeader(info, info.Name())
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	header.Name = pathInArchive // use relative paths in archive
 | |
| 
 | |
| 	err = tarWriter.WriteHeader(header)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	if info.IsDir() {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	file, err := os.Open(path)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	_, err = io.Copy(tarWriter, file)
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // ExtractTarGz reads a .tar.gz archive from the reader and extracts it into outputDirPath directory
 | |
| func ExtractTarGz(r io.Reader, outputDirPath string) error {
 | |
| 	zipReader, err := gzip.NewReader(r)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	defer zipReader.Close()
 | |
| 
 | |
| 	tarReader := tar.NewReader(zipReader)
 | |
| 
 | |
| 	for {
 | |
| 		header, err := tarReader.Next()
 | |
| 
 | |
| 		if err == io.EOF {
 | |
| 			break
 | |
| 		}
 | |
| 
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		switch header.Typeflag {
 | |
| 		case tar.TypeDir:
 | |
| 			// skip, dir will be created with a file
 | |
| 		case tar.TypeReg:
 | |
| 			p := filepath.Clean(filepath.Join(outputDirPath, header.Name))
 | |
| 			if err := os.MkdirAll(filepath.Dir(p), 0744); err != nil {
 | |
| 				return fmt.Errorf("Failed to extract dir %s", filepath.Dir(p))
 | |
| 			}
 | |
| 			outFile, err := os.Create(p)
 | |
| 			if err != nil {
 | |
| 				return fmt.Errorf("Failed to create file %s", header.Name)
 | |
| 			}
 | |
| 			if _, err := io.Copy(outFile, tarReader); err != nil {
 | |
| 				return fmt.Errorf("Failed to extract file %s", header.Name)
 | |
| 			}
 | |
| 			outFile.Close()
 | |
| 		default:
 | |
| 			return fmt.Errorf("Tar: uknown type: %v in %s",
 | |
| 				header.Typeflag,
 | |
| 				header.Name)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 |