unpack.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. package main
  2. import (
  3. "archive/tar"
  4. "compress/gzip"
  5. "io"
  6. "os"
  7. "path/filepath"
  8. "strings"
  9. "github.com/concourse/go-archive/tarfs"
  10. "github.com/google/go-containerregistry/pkg/v1"
  11. "github.com/sirupsen/logrus"
  12. pb "gopkg.in/cheggaaa/pb.v1"
  13. )
  14. const whiteoutPrefix = ".wh."
  15. func unpackImage(dest string, img v1.Image) error {
  16. layers, err := img.Layers()
  17. if err != nil {
  18. return err
  19. }
  20. written := map[string]struct{}{}
  21. removed := map[string]struct{}{}
  22. chown := os.Getuid() == 0
  23. // iterate over layers in reverse order; no need to write things files that
  24. // are modified by later layers anyway
  25. for i := len(layers) - 1; i >= 0; i-- {
  26. err := extractLayer(dest, layers[i], written, removed, chown)
  27. if err != nil {
  28. return err
  29. }
  30. }
  31. return nil
  32. }
  33. func extractLayer(dest string, layer v1.Layer, written, removed map[string]struct{}, chown bool) error {
  34. size, err := layer.Size()
  35. if err != nil {
  36. return err
  37. }
  38. digest, err := layer.Digest()
  39. if err != nil {
  40. return err
  41. }
  42. r, err := layer.Compressed()
  43. if err != nil {
  44. return err
  45. }
  46. bar := pb.New64(size).SetUnits(pb.U_BYTES)
  47. bar.Output = os.Stderr
  48. bar.Prefix(digest.Hex[0:12])
  49. bar.SetWidth(98)
  50. bar.Start()
  51. defer bar.Finish()
  52. gr, err := gzip.NewReader(bar.NewProxyReader(r))
  53. if err != nil {
  54. return err
  55. }
  56. tr := tar.NewReader(gr)
  57. for {
  58. hdr, err := tr.Next()
  59. if err == io.EOF {
  60. break
  61. }
  62. if err != nil {
  63. return err
  64. }
  65. path := filepath.Join(dest, filepath.Clean(hdr.Name))
  66. base := filepath.Base(path)
  67. dir := filepath.Dir(path)
  68. if strings.HasPrefix(base, whiteoutPrefix) {
  69. // layer has marked a file as deleted
  70. name := strings.TrimPrefix(base, whiteoutPrefix)
  71. removed[filepath.Join(dir, name)] = struct{}{}
  72. continue
  73. }
  74. if pathIsRemoved(path, removed) {
  75. // path has been removed by lower layer
  76. continue
  77. }
  78. if _, ok := written[path]; ok {
  79. // path has already been written by lower layer
  80. continue
  81. }
  82. written[path] = struct{}{}
  83. if hdr.Typeflag == tar.TypeBlock || hdr.Typeflag == tar.TypeChar {
  84. // devices can't be created in a user namespace
  85. logrus.Debugf("skipping device %s", hdr.Name)
  86. continue
  87. }
  88. if err := tarfs.ExtractEntry(hdr, dest, tr, chown); err != nil {
  89. return err
  90. }
  91. }
  92. err = gr.Close()
  93. if err != nil {
  94. return err
  95. }
  96. err = r.Close()
  97. if err != nil {
  98. return err
  99. }
  100. return nil
  101. }
  102. func pathIsRemoved(path string, removed map[string]struct{}) bool {
  103. if _, ok := removed[path]; ok {
  104. return true
  105. }
  106. // check if parent dir has been removed
  107. for wd := range removed {
  108. if strings.HasPrefix(path, wd) {
  109. return true
  110. }
  111. }
  112. return false
  113. }