changes_test.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. package archive
  2. import (
  3. "io/ioutil"
  4. "os"
  5. "os/exec"
  6. "path"
  7. "sort"
  8. "testing"
  9. "time"
  10. )
  11. func max(x, y int) int {
  12. if x >= y {
  13. return x
  14. }
  15. return y
  16. }
  17. func copyDir(src, dst string) error {
  18. cmd := exec.Command("cp", "-a", src, dst)
  19. if err := cmd.Run(); err != nil {
  20. return err
  21. }
  22. return nil
  23. }
  24. type FileType uint32
  25. const (
  26. Regular FileType = iota
  27. Dir
  28. Symlink
  29. )
  30. type FileData struct {
  31. filetype FileType
  32. path string
  33. contents string
  34. permissions os.FileMode
  35. }
  36. func createSampleDir(t *testing.T, root string) {
  37. files := []FileData{
  38. {Regular, "file1", "file1\n", 0600},
  39. {Regular, "file2", "file2\n", 0666},
  40. {Regular, "file3", "file3\n", 0404},
  41. {Regular, "file4", "file4\n", 0600},
  42. {Regular, "file5", "file5\n", 0600},
  43. {Regular, "file6", "file6\n", 0600},
  44. {Regular, "file7", "file7\n", 0600},
  45. {Dir, "dir1", "", 0740},
  46. {Regular, "dir1/file1-1", "file1-1\n", 01444},
  47. {Regular, "dir1/file1-2", "file1-2\n", 0666},
  48. {Dir, "dir2", "", 0700},
  49. {Regular, "dir2/file2-1", "file2-1\n", 0666},
  50. {Regular, "dir2/file2-2", "file2-2\n", 0666},
  51. {Dir, "dir3", "", 0700},
  52. {Regular, "dir3/file3-1", "file3-1\n", 0666},
  53. {Regular, "dir3/file3-2", "file3-2\n", 0666},
  54. {Dir, "dir4", "", 0700},
  55. {Regular, "dir4/file3-1", "file4-1\n", 0666},
  56. {Regular, "dir4/file3-2", "file4-2\n", 0666},
  57. {Symlink, "symlink1", "target1", 0666},
  58. {Symlink, "symlink2", "target2", 0666},
  59. }
  60. now := time.Now()
  61. for _, info := range files {
  62. p := path.Join(root, info.path)
  63. if info.filetype == Dir {
  64. if err := os.MkdirAll(p, info.permissions); err != nil {
  65. t.Fatal(err)
  66. }
  67. } else if info.filetype == Regular {
  68. if err := ioutil.WriteFile(p, []byte(info.contents), info.permissions); err != nil {
  69. t.Fatal(err)
  70. }
  71. } else if info.filetype == Symlink {
  72. if err := os.Symlink(info.contents, p); err != nil {
  73. t.Fatal(err)
  74. }
  75. }
  76. if info.filetype != Symlink {
  77. // Set a consistent ctime, atime for all files and dirs
  78. if err := os.Chtimes(p, now, now); err != nil {
  79. t.Fatal(err)
  80. }
  81. }
  82. }
  83. }
  84. // Create an directory, copy it, make sure we report no changes between the two
  85. func TestChangesDirsEmpty(t *testing.T) {
  86. src, err := ioutil.TempDir("", "docker-changes-test")
  87. if err != nil {
  88. t.Fatal(err)
  89. }
  90. createSampleDir(t, src)
  91. dst := src + "-copy"
  92. if err := copyDir(src, dst); err != nil {
  93. t.Fatal(err)
  94. }
  95. changes, err := ChangesDirs(dst, src)
  96. if err != nil {
  97. t.Fatal(err)
  98. }
  99. if len(changes) != 0 {
  100. t.Fatalf("Reported changes for identical dirs: %v", changes)
  101. }
  102. os.RemoveAll(src)
  103. os.RemoveAll(dst)
  104. }
  105. func mutateSampleDir(t *testing.T, root string) {
  106. // Remove a regular file
  107. if err := os.RemoveAll(path.Join(root, "file1")); err != nil {
  108. t.Fatal(err)
  109. }
  110. // Remove a directory
  111. if err := os.RemoveAll(path.Join(root, "dir1")); err != nil {
  112. t.Fatal(err)
  113. }
  114. // Remove a symlink
  115. if err := os.RemoveAll(path.Join(root, "symlink1")); err != nil {
  116. t.Fatal(err)
  117. }
  118. // Rewrite a file
  119. if err := ioutil.WriteFile(path.Join(root, "file2"), []byte("fileNN\n"), 0777); err != nil {
  120. t.Fatal(err)
  121. }
  122. // Replace a file
  123. if err := os.RemoveAll(path.Join(root, "file3")); err != nil {
  124. t.Fatal(err)
  125. }
  126. if err := ioutil.WriteFile(path.Join(root, "file3"), []byte("fileMM\n"), 0404); err != nil {
  127. t.Fatal(err)
  128. }
  129. // Touch file
  130. if err := os.Chtimes(path.Join(root, "file4"), time.Now().Add(time.Second), time.Now().Add(time.Second)); err != nil {
  131. t.Fatal(err)
  132. }
  133. // Replace file with dir
  134. if err := os.RemoveAll(path.Join(root, "file5")); err != nil {
  135. t.Fatal(err)
  136. }
  137. if err := os.MkdirAll(path.Join(root, "file5"), 0666); err != nil {
  138. t.Fatal(err)
  139. }
  140. // Create new file
  141. if err := ioutil.WriteFile(path.Join(root, "filenew"), []byte("filenew\n"), 0777); err != nil {
  142. t.Fatal(err)
  143. }
  144. // Create new dir
  145. if err := os.MkdirAll(path.Join(root, "dirnew"), 0766); err != nil {
  146. t.Fatal(err)
  147. }
  148. // Create a new symlink
  149. if err := os.Symlink("targetnew", path.Join(root, "symlinknew")); err != nil {
  150. t.Fatal(err)
  151. }
  152. // Change a symlink
  153. if err := os.RemoveAll(path.Join(root, "symlink2")); err != nil {
  154. t.Fatal(err)
  155. }
  156. if err := os.Symlink("target2change", path.Join(root, "symlink2")); err != nil {
  157. t.Fatal(err)
  158. }
  159. // Replace dir with file
  160. if err := os.RemoveAll(path.Join(root, "dir2")); err != nil {
  161. t.Fatal(err)
  162. }
  163. if err := ioutil.WriteFile(path.Join(root, "dir2"), []byte("dir2\n"), 0777); err != nil {
  164. t.Fatal(err)
  165. }
  166. // Touch dir
  167. if err := os.Chtimes(path.Join(root, "dir3"), time.Now().Add(time.Second), time.Now().Add(time.Second)); err != nil {
  168. t.Fatal(err)
  169. }
  170. }
  171. func TestChangesDirsMutated(t *testing.T) {
  172. src, err := ioutil.TempDir("", "docker-changes-test")
  173. if err != nil {
  174. t.Fatal(err)
  175. }
  176. createSampleDir(t, src)
  177. dst := src + "-copy"
  178. if err := copyDir(src, dst); err != nil {
  179. t.Fatal(err)
  180. }
  181. defer os.RemoveAll(src)
  182. defer os.RemoveAll(dst)
  183. mutateSampleDir(t, dst)
  184. changes, err := ChangesDirs(dst, src)
  185. if err != nil {
  186. t.Fatal(err)
  187. }
  188. sort.Sort(changesByPath(changes))
  189. expectedChanges := []Change{
  190. {"/dir1", ChangeDelete},
  191. {"/dir2", ChangeModify},
  192. {"/dir3", ChangeModify},
  193. {"/dirnew", ChangeAdd},
  194. {"/file1", ChangeDelete},
  195. {"/file2", ChangeModify},
  196. {"/file3", ChangeModify},
  197. {"/file4", ChangeModify},
  198. {"/file5", ChangeModify},
  199. {"/filenew", ChangeAdd},
  200. {"/symlink1", ChangeDelete},
  201. {"/symlink2", ChangeModify},
  202. {"/symlinknew", ChangeAdd},
  203. }
  204. for i := 0; i < max(len(changes), len(expectedChanges)); i++ {
  205. if i >= len(expectedChanges) {
  206. t.Fatalf("unexpected change %s\n", changes[i].String())
  207. }
  208. if i >= len(changes) {
  209. t.Fatalf("no change for expected change %s\n", expectedChanges[i].String())
  210. }
  211. if changes[i].Path == expectedChanges[i].Path {
  212. if changes[i] != expectedChanges[i] {
  213. t.Fatalf("Wrong change for %s, expected %s, got %s\n", changes[i].Path, changes[i].String(), expectedChanges[i].String())
  214. }
  215. } else if changes[i].Path < expectedChanges[i].Path {
  216. t.Fatalf("unexpected change %s\n", changes[i].String())
  217. } else {
  218. t.Fatalf("no change for expected change %s != %s\n", expectedChanges[i].String(), changes[i].String())
  219. }
  220. }
  221. }
  222. func TestApplyLayer(t *testing.T) {
  223. src, err := ioutil.TempDir("", "docker-changes-test")
  224. if err != nil {
  225. t.Fatal(err)
  226. }
  227. createSampleDir(t, src)
  228. defer os.RemoveAll(src)
  229. dst := src + "-copy"
  230. if err := copyDir(src, dst); err != nil {
  231. t.Fatal(err)
  232. }
  233. mutateSampleDir(t, dst)
  234. defer os.RemoveAll(dst)
  235. changes, err := ChangesDirs(dst, src)
  236. if err != nil {
  237. t.Fatal(err)
  238. }
  239. layer, err := ExportChanges(dst, changes)
  240. if err != nil {
  241. t.Fatal(err)
  242. }
  243. layerCopy, err := NewTempArchive(layer, "")
  244. if err != nil {
  245. t.Fatal(err)
  246. }
  247. if _, err := ApplyLayer(src, layerCopy); err != nil {
  248. t.Fatal(err)
  249. }
  250. changes2, err := ChangesDirs(src, dst)
  251. if err != nil {
  252. t.Fatal(err)
  253. }
  254. if len(changes2) != 0 {
  255. t.Fatalf("Unexpected differences after reapplying mutation: %v", changes2)
  256. }
  257. }