archive_unix_test.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. // +build !windows
  2. package archive
  3. import (
  4. "bytes"
  5. "fmt"
  6. "io/ioutil"
  7. "os"
  8. "path/filepath"
  9. "syscall"
  10. "testing"
  11. "github.com/docker/docker/pkg/system"
  12. )
  13. func TestCanonicalTarNameForPath(t *testing.T) {
  14. cases := []struct{ in, expected string }{
  15. {"foo", "foo"},
  16. {"foo/bar", "foo/bar"},
  17. {"foo/dir/", "foo/dir/"},
  18. }
  19. for _, v := range cases {
  20. if out, err := CanonicalTarNameForPath(v.in); err != nil {
  21. t.Fatalf("cannot get canonical name for path: %s: %v", v.in, err)
  22. } else if out != v.expected {
  23. t.Fatalf("wrong canonical tar name. expected:%s got:%s", v.expected, out)
  24. }
  25. }
  26. }
  27. func TestCanonicalTarName(t *testing.T) {
  28. cases := []struct {
  29. in string
  30. isDir bool
  31. expected string
  32. }{
  33. {"foo", false, "foo"},
  34. {"foo", true, "foo/"},
  35. {"foo/bar", false, "foo/bar"},
  36. {"foo/bar", true, "foo/bar/"},
  37. }
  38. for _, v := range cases {
  39. if out, err := canonicalTarName(v.in, v.isDir); err != nil {
  40. t.Fatalf("cannot get canonical name for path: %s: %v", v.in, err)
  41. } else if out != v.expected {
  42. t.Fatalf("wrong canonical tar name. expected:%s got:%s", v.expected, out)
  43. }
  44. }
  45. }
  46. func TestChmodTarEntry(t *testing.T) {
  47. cases := []struct {
  48. in, expected os.FileMode
  49. }{
  50. {0000, 0000},
  51. {0777, 0777},
  52. {0644, 0644},
  53. {0755, 0755},
  54. {0444, 0444},
  55. }
  56. for _, v := range cases {
  57. if out := chmodTarEntry(v.in); out != v.expected {
  58. t.Fatalf("wrong chmod. expected:%v got:%v", v.expected, out)
  59. }
  60. }
  61. }
  62. func TestTarWithHardLink(t *testing.T) {
  63. origin, err := ioutil.TempDir("", "docker-test-tar-hardlink")
  64. if err != nil {
  65. t.Fatal(err)
  66. }
  67. defer os.RemoveAll(origin)
  68. if err := ioutil.WriteFile(filepath.Join(origin, "1"), []byte("hello world"), 0700); err != nil {
  69. t.Fatal(err)
  70. }
  71. if err := os.Link(filepath.Join(origin, "1"), filepath.Join(origin, "2")); err != nil {
  72. t.Fatal(err)
  73. }
  74. var i1, i2 uint64
  75. if i1, err = getNlink(filepath.Join(origin, "1")); err != nil {
  76. t.Fatal(err)
  77. }
  78. // sanity check that we can hardlink
  79. if i1 != 2 {
  80. t.Skipf("skipping since hardlinks don't work here; expected 2 links, got %d", i1)
  81. }
  82. dest, err := ioutil.TempDir("", "docker-test-tar-hardlink-dest")
  83. if err != nil {
  84. t.Fatal(err)
  85. }
  86. defer os.RemoveAll(dest)
  87. // we'll do this in two steps to separate failure
  88. fh, err := Tar(origin, Uncompressed)
  89. if err != nil {
  90. t.Fatal(err)
  91. }
  92. // ensure we can read the whole thing with no error, before writing back out
  93. buf, err := ioutil.ReadAll(fh)
  94. if err != nil {
  95. t.Fatal(err)
  96. }
  97. bRdr := bytes.NewReader(buf)
  98. err = Untar(bRdr, dest, &TarOptions{Compression: Uncompressed})
  99. if err != nil {
  100. t.Fatal(err)
  101. }
  102. if i1, err = getInode(filepath.Join(dest, "1")); err != nil {
  103. t.Fatal(err)
  104. }
  105. if i2, err = getInode(filepath.Join(dest, "2")); err != nil {
  106. t.Fatal(err)
  107. }
  108. if i1 != i2 {
  109. t.Errorf("expected matching inodes, but got %d and %d", i1, i2)
  110. }
  111. }
  112. func getNlink(path string) (uint64, error) {
  113. stat, err := os.Stat(path)
  114. if err != nil {
  115. return 0, err
  116. }
  117. statT, ok := stat.Sys().(*syscall.Stat_t)
  118. if !ok {
  119. return 0, fmt.Errorf("expected type *syscall.Stat_t, got %t", stat.Sys())
  120. }
  121. // We need this conversion on ARM64
  122. return uint64(statT.Nlink), nil
  123. }
  124. func getInode(path string) (uint64, error) {
  125. stat, err := os.Stat(path)
  126. if err != nil {
  127. return 0, err
  128. }
  129. statT, ok := stat.Sys().(*syscall.Stat_t)
  130. if !ok {
  131. return 0, fmt.Errorf("expected type *syscall.Stat_t, got %t", stat.Sys())
  132. }
  133. return statT.Ino, nil
  134. }
  135. func TestTarWithBlockCharFifo(t *testing.T) {
  136. origin, err := ioutil.TempDir("", "docker-test-tar-hardlink")
  137. if err != nil {
  138. t.Fatal(err)
  139. }
  140. defer os.RemoveAll(origin)
  141. if err := ioutil.WriteFile(filepath.Join(origin, "1"), []byte("hello world"), 0700); err != nil {
  142. t.Fatal(err)
  143. }
  144. if err := system.Mknod(filepath.Join(origin, "2"), syscall.S_IFBLK, int(system.Mkdev(int64(12), int64(5)))); err != nil {
  145. t.Fatal(err)
  146. }
  147. if err := system.Mknod(filepath.Join(origin, "3"), syscall.S_IFCHR, int(system.Mkdev(int64(12), int64(5)))); err != nil {
  148. t.Fatal(err)
  149. }
  150. if err := system.Mknod(filepath.Join(origin, "4"), syscall.S_IFIFO, int(system.Mkdev(int64(12), int64(5)))); err != nil {
  151. t.Fatal(err)
  152. }
  153. dest, err := ioutil.TempDir("", "docker-test-tar-hardlink-dest")
  154. if err != nil {
  155. t.Fatal(err)
  156. }
  157. defer os.RemoveAll(dest)
  158. // we'll do this in two steps to separate failure
  159. fh, err := Tar(origin, Uncompressed)
  160. if err != nil {
  161. t.Fatal(err)
  162. }
  163. // ensure we can read the whole thing with no error, before writing back out
  164. buf, err := ioutil.ReadAll(fh)
  165. if err != nil {
  166. t.Fatal(err)
  167. }
  168. bRdr := bytes.NewReader(buf)
  169. err = Untar(bRdr, dest, &TarOptions{Compression: Uncompressed})
  170. if err != nil {
  171. t.Fatal(err)
  172. }
  173. changes, err := ChangesDirs(origin, dest)
  174. if err != nil {
  175. t.Fatal(err)
  176. }
  177. if len(changes) > 0 {
  178. t.Fatalf("Tar with special device (block, char, fifo) should keep them (recreate them when untar) : %v", changes)
  179. }
  180. }
  181. // TestTarUntarWithXattr is Unix as Lsetxattr is not supported on Windows
  182. func TestTarUntarWithXattr(t *testing.T) {
  183. origin, err := ioutil.TempDir("", "docker-test-untar-origin")
  184. if err != nil {
  185. t.Fatal(err)
  186. }
  187. defer os.RemoveAll(origin)
  188. if err := ioutil.WriteFile(filepath.Join(origin, "1"), []byte("hello world"), 0700); err != nil {
  189. t.Fatal(err)
  190. }
  191. if err := ioutil.WriteFile(filepath.Join(origin, "2"), []byte("welcome!"), 0700); err != nil {
  192. t.Fatal(err)
  193. }
  194. if err := ioutil.WriteFile(filepath.Join(origin, "3"), []byte("will be ignored"), 0700); err != nil {
  195. t.Fatal(err)
  196. }
  197. if err := system.Lsetxattr(filepath.Join(origin, "2"), "security.capability", []byte{0x00}, 0); err != nil {
  198. t.Fatal(err)
  199. }
  200. for _, c := range []Compression{
  201. Uncompressed,
  202. Gzip,
  203. } {
  204. changes, err := tarUntar(t, origin, &TarOptions{
  205. Compression: c,
  206. ExcludePatterns: []string{"3"},
  207. })
  208. if err != nil {
  209. t.Fatalf("Error tar/untar for compression %s: %s", c.Extension(), err)
  210. }
  211. if len(changes) != 1 || changes[0].Path != "/3" {
  212. t.Fatalf("Unexpected differences after tarUntar: %v", changes)
  213. }
  214. capability, _ := system.Lgetxattr(filepath.Join(origin, "2"), "security.capability")
  215. if capability == nil && capability[0] != 0x00 {
  216. t.Fatalf("Untar should have kept the 'security.capability' xattr.")
  217. }
  218. }
  219. }