-
Notifications
You must be signed in to change notification settings - Fork 36
/
Copy pathunarchive.go
102 lines (83 loc) · 2.33 KB
/
unarchive.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
// Functions for expanding gzipped tarballs
package main
import (
"archive/tar"
"compress/gzip"
"fmt"
"github.com/pkg/errors"
"github.com/sirupsen/logrus"
"io"
"net/http"
"os"
"path/filepath"
)
// Ensures that a given file is gzip-encoded
func ensureGzip(file io.Reader) error {
buff := make([]byte, 512) // content-type length
_, err := file.Read(buff)
if err != nil {
return errors.Wrap(err, "unable to detect encoding")
}
if filetype := http.DetectContentType(buff); filetype != "application/x-gzip" {
return fmt.Errorf("file not gzip encoded")
}
return nil
}
// Extract a tarball from the src into dest
func extractTgz(src, dest string) error {
logrus.Debugf("Expanding %s to %s", src, dest)
tgzFile, err := os.Open(src)
if err != nil {
return errors.Wrap(err, "unable to open source file")
}
defer tgzFile.Close()
err = ensureGzip(tgzFile)
if err != nil {
return errors.Wrap(err, "unable to make gzip reader")
}
tgzFile.Seek(0, 0)
uncompressedStream, err := gzip.NewReader(tgzFile)
if err != nil {
errors.Wrap(err, "unable to make gzip reader")
}
defer uncompressedStream.Close()
tarReader := tar.NewReader(uncompressedStream)
if _, err := os.Stat(dest); os.IsNotExist(err) {
if err := os.Mkdir(dest, 0755); err != nil {
return errors.Wrap(err, "unable to create target directory")
}
}
for {
header, err := tarReader.Next()
switch {
case err == io.EOF:
return nil
case err != nil:
return errors.Wrap(err, "unable to extract gzipped tarfile")
case header == nil:
continue // phantom file case
}
targetPath := filepath.Join(dest, header.Name)
switch header.Typeflag {
case tar.TypeSymlink:
if err := os.Symlink(header.Linkname, targetPath); err != nil {
return errors.Wrap(err, "unable to create symlink from tar")
}
case tar.TypeDir:
if _, err := os.Stat(targetPath); os.IsNotExist(err) {
if err := os.Mkdir(targetPath, 0755); err != nil {
return errors.Wrap(err, "unable to create dir from tar")
}
}
case tar.TypeReg:
outFile, err := os.OpenFile(targetPath, os.O_CREATE|os.O_RDWR, os.FileMode(header.Mode))
if err != nil {
return errors.Wrap(err, "unable to create file from tar")
}
if _, err := io.Copy(outFile, tarReader); err != nil {
return errors.Wrap(err, "unable to populate file from tar")
}
outFile.Close()
}
}
}