tarsum_test.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. package tarsum
  2. import (
  3. "bytes"
  4. "crypto/rand"
  5. "fmt"
  6. "io"
  7. "io/ioutil"
  8. "os"
  9. "testing"
  10. "github.com/docker/docker/vendor/src/code.google.com/p/go/src/pkg/archive/tar"
  11. )
  12. type testLayer struct {
  13. filename string
  14. options *sizedOptions
  15. jsonfile string
  16. gzip bool
  17. tarsum string
  18. version Version
  19. }
  20. var testLayers = []testLayer{
  21. {
  22. filename: "testdata/46af0962ab5afeb5ce6740d4d91652e69206fc991fd5328c1a94d364ad00e457/layer.tar",
  23. jsonfile: "testdata/46af0962ab5afeb5ce6740d4d91652e69206fc991fd5328c1a94d364ad00e457/json",
  24. version: Version0,
  25. tarsum: "tarsum+sha256:e58fcf7418d4390dec8e8fb69d88c06ec07039d651fedd3aa72af9972e7d046b"},
  26. {
  27. filename: "testdata/46af0962ab5afeb5ce6740d4d91652e69206fc991fd5328c1a94d364ad00e457/layer.tar",
  28. jsonfile: "testdata/46af0962ab5afeb5ce6740d4d91652e69206fc991fd5328c1a94d364ad00e457/json",
  29. version: VersionDev,
  30. tarsum: "tarsum.dev+sha256:486b86e25c4db4551228154848bc4663b15dd95784b1588980f4ba1cb42e83e9"},
  31. {
  32. filename: "testdata/46af0962ab5afeb5ce6740d4d91652e69206fc991fd5328c1a94d364ad00e457/layer.tar",
  33. jsonfile: "testdata/46af0962ab5afeb5ce6740d4d91652e69206fc991fd5328c1a94d364ad00e457/json",
  34. gzip: true,
  35. tarsum: "tarsum+sha256:e58fcf7418d4390dec8e8fb69d88c06ec07039d651fedd3aa72af9972e7d046b"},
  36. {
  37. // Tests existing version of TarSum when xattrs are present
  38. filename: "testdata/xattr/layer.tar",
  39. jsonfile: "testdata/xattr/json",
  40. version: Version0,
  41. tarsum: "tarsum+sha256:e86f81a4d552f13039b1396ed03ca968ea9717581f9577ef1876ea6ff9b38c98"},
  42. {
  43. // Tests next version of TarSum when xattrs are present
  44. filename: "testdata/xattr/layer.tar",
  45. jsonfile: "testdata/xattr/json",
  46. version: VersionDev,
  47. tarsum: "tarsum.dev+sha256:6235cd3a2afb7501bac541772a3d61a3634e95bc90bb39a4676e2cb98d08390d"},
  48. {
  49. filename: "testdata/511136ea3c5a64f264b78b5433614aec563103b4d4702f3ba7d4d2698e22c158/layer.tar",
  50. jsonfile: "testdata/511136ea3c5a64f264b78b5433614aec563103b4d4702f3ba7d4d2698e22c158/json",
  51. tarsum: "tarsum+sha256:ac672ee85da9ab7f9667ae3c32841d3e42f33cc52c273c23341dabba1c8b0c8b"},
  52. {
  53. options: &sizedOptions{1, 1024 * 1024, false, false}, // a 1mb file (in memory)
  54. tarsum: "tarsum+sha256:8bf12d7e67c51ee2e8306cba569398b1b9f419969521a12ffb9d8875e8836738"},
  55. }
  56. type sizedOptions struct {
  57. num int64
  58. size int64
  59. isRand bool
  60. realFile bool
  61. }
  62. // make a tar:
  63. // * num is the number of files the tar should have
  64. // * size is the bytes per file
  65. // * isRand is whether the contents of the files should be a random chunk (otherwise it's all zeros)
  66. // * realFile will write to a TempFile, instead of an in memory buffer
  67. func sizedTar(opts sizedOptions) io.Reader {
  68. var (
  69. fh io.ReadWriter
  70. err error
  71. )
  72. if opts.realFile {
  73. fh, err = ioutil.TempFile("", "tarsum")
  74. if err != nil {
  75. return nil
  76. }
  77. } else {
  78. fh = bytes.NewBuffer([]byte{})
  79. }
  80. tarW := tar.NewWriter(fh)
  81. for i := int64(0); i < opts.num; i++ {
  82. err := tarW.WriteHeader(&tar.Header{
  83. Name: fmt.Sprintf("/testdata%d", i),
  84. Mode: 0755,
  85. Uid: 0,
  86. Gid: 0,
  87. Size: opts.size,
  88. })
  89. if err != nil {
  90. return nil
  91. }
  92. var rBuf []byte
  93. if opts.isRand {
  94. rBuf = make([]byte, 8)
  95. _, err = rand.Read(rBuf)
  96. if err != nil {
  97. return nil
  98. }
  99. } else {
  100. rBuf = []byte{0, 0, 0, 0, 0, 0, 0, 0}
  101. }
  102. for i := int64(0); i < opts.size/int64(8); i++ {
  103. tarW.Write(rBuf)
  104. }
  105. }
  106. return fh
  107. }
  108. func TestTarSums(t *testing.T) {
  109. for _, layer := range testLayers {
  110. var (
  111. fh io.Reader
  112. err error
  113. )
  114. if len(layer.filename) > 0 {
  115. fh, err = os.Open(layer.filename)
  116. if err != nil {
  117. t.Errorf("failed to open %s: %s", layer.filename, err)
  118. continue
  119. }
  120. } else if layer.options != nil {
  121. fh = sizedTar(*layer.options)
  122. } else {
  123. // What else is there to test?
  124. t.Errorf("what to do with %#v", layer)
  125. continue
  126. }
  127. if file, ok := fh.(*os.File); ok {
  128. defer file.Close()
  129. }
  130. // double negatives!
  131. ts, err := NewTarSum(fh, !layer.gzip, layer.version)
  132. if err != nil {
  133. t.Errorf("%q :: %q", err, layer.filename)
  134. continue
  135. }
  136. _, err = io.Copy(ioutil.Discard, ts)
  137. if err != nil {
  138. t.Errorf("failed to copy from %s: %s", layer.filename, err)
  139. continue
  140. }
  141. var gotSum string
  142. if len(layer.jsonfile) > 0 {
  143. jfh, err := os.Open(layer.jsonfile)
  144. if err != nil {
  145. t.Errorf("failed to open %s: %s", layer.jsonfile, err)
  146. continue
  147. }
  148. buf, err := ioutil.ReadAll(jfh)
  149. if err != nil {
  150. t.Errorf("failed to readAll %s: %s", layer.jsonfile, err)
  151. continue
  152. }
  153. gotSum = ts.Sum(buf)
  154. } else {
  155. gotSum = ts.Sum(nil)
  156. }
  157. if layer.tarsum != gotSum {
  158. t.Errorf("expecting [%s], but got [%s]", layer.tarsum, gotSum)
  159. }
  160. }
  161. }
  162. func Benchmark9kTar(b *testing.B) {
  163. buf := bytes.NewBuffer([]byte{})
  164. fh, err := os.Open("testdata/46af0962ab5afeb5ce6740d4d91652e69206fc991fd5328c1a94d364ad00e457/layer.tar")
  165. if err != nil {
  166. b.Error(err)
  167. return
  168. }
  169. n, err := io.Copy(buf, fh)
  170. fh.Close()
  171. b.SetBytes(n)
  172. b.ResetTimer()
  173. for i := 0; i < b.N; i++ {
  174. ts, err := NewTarSum(buf, true, Version0)
  175. if err != nil {
  176. b.Error(err)
  177. return
  178. }
  179. io.Copy(ioutil.Discard, ts)
  180. ts.Sum(nil)
  181. }
  182. }
  183. func Benchmark9kTarGzip(b *testing.B) {
  184. buf := bytes.NewBuffer([]byte{})
  185. fh, err := os.Open("testdata/46af0962ab5afeb5ce6740d4d91652e69206fc991fd5328c1a94d364ad00e457/layer.tar")
  186. if err != nil {
  187. b.Error(err)
  188. return
  189. }
  190. n, err := io.Copy(buf, fh)
  191. fh.Close()
  192. b.SetBytes(n)
  193. b.ResetTimer()
  194. for i := 0; i < b.N; i++ {
  195. ts, err := NewTarSum(buf, false, Version0)
  196. if err != nil {
  197. b.Error(err)
  198. return
  199. }
  200. io.Copy(ioutil.Discard, ts)
  201. ts.Sum(nil)
  202. }
  203. }
  204. // this is a single big file in the tar archive
  205. func Benchmark1mbSingleFileTar(b *testing.B) {
  206. benchmarkTar(b, sizedOptions{1, 1024 * 1024, true, true}, false)
  207. }
  208. // this is a single big file in the tar archive
  209. func Benchmark1mbSingleFileTarGzip(b *testing.B) {
  210. benchmarkTar(b, sizedOptions{1, 1024 * 1024, true, true}, true)
  211. }
  212. // this is 1024 1k files in the tar archive
  213. func Benchmark1kFilesTar(b *testing.B) {
  214. benchmarkTar(b, sizedOptions{1024, 1024, true, true}, false)
  215. }
  216. // this is 1024 1k files in the tar archive
  217. func Benchmark1kFilesTarGzip(b *testing.B) {
  218. benchmarkTar(b, sizedOptions{1024, 1024, true, true}, true)
  219. }
  220. func benchmarkTar(b *testing.B, opts sizedOptions, isGzip bool) {
  221. var fh *os.File
  222. tarReader := sizedTar(opts)
  223. if br, ok := tarReader.(*os.File); ok {
  224. fh = br
  225. }
  226. defer os.Remove(fh.Name())
  227. defer fh.Close()
  228. b.SetBytes(opts.size * opts.num)
  229. b.ResetTimer()
  230. for i := 0; i < b.N; i++ {
  231. ts, err := NewTarSum(fh, !isGzip, Version0)
  232. if err != nil {
  233. b.Error(err)
  234. return
  235. }
  236. io.Copy(ioutil.Discard, ts)
  237. ts.Sum(nil)
  238. fh.Seek(0, 0)
  239. }
  240. }