tar.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. // License: GPLv3 Copyright: 2023, Kovid Goyal, <kovid at kovidgoyal.net>
  2. package utils
  3. import (
  4. "archive/tar"
  5. "errors"
  6. "fmt"
  7. "io"
  8. "io/fs"
  9. "os"
  10. "path/filepath"
  11. "strings"
  12. )
  13. var _ = fmt.Print
  14. type TarExtractOptions struct {
  15. DontPreservePermissions bool
  16. }
  17. func ExtractAllFromTar(tr *tar.Reader, dest_path string, optss ...TarExtractOptions) (count int, err error) {
  18. opts := TarExtractOptions{}
  19. if len(optss) > 0 {
  20. opts = optss[0]
  21. }
  22. dest_path, err = filepath.Abs(dest_path)
  23. if err != nil {
  24. return
  25. }
  26. mode := func(hdr *tar.Header) fs.FileMode {
  27. return fs.FileMode(hdr.Mode) & (fs.ModePerm | fs.ModeSetgid | fs.ModeSetuid | fs.ModeSticky)
  28. }
  29. set_metadata := func(chmod func(mode fs.FileMode) error, hdr *tar.Header) (err error) {
  30. if !opts.DontPreservePermissions && chmod != nil {
  31. perms := mode(hdr)
  32. if err = chmod(perms); err != nil {
  33. return err
  34. }
  35. }
  36. count++
  37. return
  38. }
  39. for {
  40. var hdr *tar.Header
  41. hdr, err = tr.Next()
  42. if errors.Is(err, io.EOF) {
  43. err = nil
  44. break
  45. }
  46. if err != nil {
  47. return count, err
  48. }
  49. dest := hdr.Name
  50. dest = strings.TrimLeft(dest, "/")
  51. if !filepath.IsLocal(dest) {
  52. continue
  53. }
  54. dest = filepath.Join(dest_path, dest)
  55. switch hdr.Typeflag {
  56. case tar.TypeDir:
  57. err = os.MkdirAll(dest, 0o700)
  58. if err != nil {
  59. return
  60. }
  61. if err = set_metadata(func(m fs.FileMode) error { return os.Chmod(dest, m) }, hdr); err != nil {
  62. return
  63. }
  64. case tar.TypeReg, tar.TypeRegA:
  65. var d *os.File
  66. if err = os.MkdirAll(filepath.Dir(dest), 0o700); err != nil {
  67. return
  68. }
  69. if d, err = os.Create(dest); err != nil {
  70. return
  71. }
  72. err = set_metadata(d.Chmod, hdr)
  73. if err == nil {
  74. _, err = io.Copy(d, tr)
  75. }
  76. d.Close()
  77. if err != nil {
  78. return
  79. }
  80. case tar.TypeLink:
  81. if err = os.MkdirAll(filepath.Dir(dest), 0o700); err != nil {
  82. return
  83. }
  84. if err = os.Link(hdr.Linkname, dest); err != nil {
  85. return
  86. }
  87. if err = set_metadata(func(m fs.FileMode) error { return os.Chmod(dest, m) }, hdr); err != nil {
  88. return
  89. }
  90. case tar.TypeSymlink:
  91. if err = os.MkdirAll(filepath.Dir(dest), 0o700); err != nil {
  92. return
  93. }
  94. if err = os.Symlink(hdr.Linkname, dest); err != nil {
  95. return
  96. }
  97. if err = set_metadata(nil, hdr); err != nil {
  98. return
  99. }
  100. }
  101. }
  102. return
  103. }