fs_test.go 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. package image // import "github.com/docker/docker/image"
  2. import (
  3. "crypto/rand"
  4. "crypto/sha256"
  5. "encoding/hex"
  6. "errors"
  7. "os"
  8. "path/filepath"
  9. "testing"
  10. "github.com/opencontainers/go-digest"
  11. "gotest.tools/v3/assert"
  12. is "gotest.tools/v3/assert/cmp"
  13. )
  14. func defaultFSStoreBackend(t *testing.T) (StoreBackend, func()) {
  15. tmpdir, err := os.MkdirTemp("", "images-fs-store")
  16. assert.Check(t, err)
  17. fsBackend, err := NewFSStoreBackend(tmpdir)
  18. assert.Check(t, err)
  19. return fsBackend, func() { os.RemoveAll(tmpdir) }
  20. }
  21. func TestFSGetInvalidData(t *testing.T) {
  22. store, cleanup := defaultFSStoreBackend(t)
  23. defer cleanup()
  24. dgst, err := store.Set([]byte("foobar"))
  25. assert.Check(t, err)
  26. err = os.WriteFile(filepath.Join(store.(*fs).root, contentDirName, string(dgst.Algorithm()), dgst.Encoded()), []byte("foobar2"), 0o600)
  27. assert.Check(t, err)
  28. _, err = store.Get(dgst)
  29. assert.Check(t, is.ErrorContains(err, "failed to verify"))
  30. }
  31. func TestFSInvalidSet(t *testing.T) {
  32. store, cleanup := defaultFSStoreBackend(t)
  33. defer cleanup()
  34. id := digest.FromBytes([]byte("foobar"))
  35. err := os.Mkdir(filepath.Join(store.(*fs).root, contentDirName, string(id.Algorithm()), id.Encoded()), 0o700)
  36. assert.Check(t, err)
  37. _, err = store.Set([]byte("foobar"))
  38. assert.Check(t, is.ErrorContains(err, "failed to write digest data"))
  39. }
  40. func TestFSInvalidRoot(t *testing.T) {
  41. tmpdir, err := os.MkdirTemp("", "images-fs-store")
  42. assert.Check(t, err)
  43. defer os.RemoveAll(tmpdir)
  44. tcases := []struct {
  45. root, invalidFile string
  46. }{
  47. {"root", "root"},
  48. {"root", "root/content"},
  49. {"root", "root/metadata"},
  50. }
  51. for _, tc := range tcases {
  52. root := filepath.Join(tmpdir, tc.root)
  53. filePath := filepath.Join(tmpdir, tc.invalidFile)
  54. err := os.MkdirAll(filepath.Dir(filePath), 0o700)
  55. assert.Check(t, err)
  56. f, err := os.Create(filePath)
  57. assert.Check(t, err)
  58. f.Close()
  59. _, err = NewFSStoreBackend(root)
  60. assert.Check(t, is.ErrorContains(err, "failed to create storage backend"))
  61. os.RemoveAll(root)
  62. }
  63. }
  64. func TestFSMetadataGetSet(t *testing.T) {
  65. store, cleanup := defaultFSStoreBackend(t)
  66. defer cleanup()
  67. id, err := store.Set([]byte("foo"))
  68. assert.Check(t, err)
  69. id2, err := store.Set([]byte("bar"))
  70. assert.Check(t, err)
  71. tcases := []struct {
  72. id digest.Digest
  73. key string
  74. value []byte
  75. }{
  76. {id, "tkey", []byte("tval1")},
  77. {id, "tkey2", []byte("tval2")},
  78. {id2, "tkey", []byte("tval3")},
  79. }
  80. for _, tc := range tcases {
  81. err = store.SetMetadata(tc.id, tc.key, tc.value)
  82. assert.Check(t, err)
  83. actual, err := store.GetMetadata(tc.id, tc.key)
  84. assert.Check(t, err)
  85. assert.Check(t, is.DeepEqual(tc.value, actual))
  86. }
  87. _, err = store.GetMetadata(id2, "tkey2")
  88. assert.Check(t, is.ErrorContains(err, "failed to read metadata"))
  89. id3 := digest.FromBytes([]byte("baz"))
  90. err = store.SetMetadata(id3, "tkey", []byte("tval"))
  91. assert.Check(t, is.ErrorContains(err, "failed to get digest"))
  92. _, err = store.GetMetadata(id3, "tkey")
  93. assert.Check(t, is.ErrorContains(err, "failed to get digest"))
  94. }
  95. func TestFSInvalidWalker(t *testing.T) {
  96. store, cleanup := defaultFSStoreBackend(t)
  97. defer cleanup()
  98. fooID, err := store.Set([]byte("foo"))
  99. assert.Check(t, err)
  100. err = os.WriteFile(filepath.Join(store.(*fs).root, contentDirName, "sha256/foobar"), []byte("foobar"), 0o600)
  101. assert.Check(t, err)
  102. n := 0
  103. err = store.Walk(func(id digest.Digest) error {
  104. assert.Check(t, is.Equal(fooID, id))
  105. n++
  106. return nil
  107. })
  108. assert.Check(t, err)
  109. assert.Check(t, is.Equal(1, n))
  110. }
  111. func TestFSGetSet(t *testing.T) {
  112. store, cleanup := defaultFSStoreBackend(t)
  113. defer cleanup()
  114. type tcase struct {
  115. input []byte
  116. expected digest.Digest
  117. }
  118. tcases := []tcase{
  119. {[]byte("foobar"), digest.Digest("sha256:c3ab8ff13720e8ad9047dd39466b3c8974e592c2fa383d4a3960714caef0c4f2")},
  120. }
  121. randomInput := make([]byte, 8*1024)
  122. _, err := rand.Read(randomInput)
  123. assert.Check(t, err)
  124. // skipping use of digest pkg because it is used by the implementation
  125. h := sha256.New()
  126. _, err = h.Write(randomInput)
  127. assert.Check(t, err)
  128. tcases = append(tcases, tcase{
  129. input: randomInput,
  130. expected: digest.Digest("sha256:" + hex.EncodeToString(h.Sum(nil))),
  131. })
  132. for _, tc := range tcases {
  133. id, err := store.Set(tc.input)
  134. assert.Check(t, err)
  135. assert.Check(t, is.Equal(tc.expected, id))
  136. }
  137. for _, tc := range tcases {
  138. data, err := store.Get(tc.expected)
  139. assert.Check(t, err)
  140. assert.Check(t, is.DeepEqual(tc.input, data))
  141. }
  142. }
  143. func TestFSGetUnsetKey(t *testing.T) {
  144. store, cleanup := defaultFSStoreBackend(t)
  145. defer cleanup()
  146. for _, key := range []digest.Digest{"foobar:abc", "sha256:abc", "sha256:c3ab8ff13720e8ad9047dd39466b3c8974e592c2fa383d4a3960714caef0c4f2a"} {
  147. _, err := store.Get(key)
  148. assert.Check(t, is.ErrorContains(err, "failed to get digest"))
  149. }
  150. }
  151. func TestFSGetEmptyData(t *testing.T) {
  152. store, cleanup := defaultFSStoreBackend(t)
  153. defer cleanup()
  154. for _, emptyData := range [][]byte{nil, {}} {
  155. _, err := store.Set(emptyData)
  156. assert.Check(t, is.ErrorContains(err, "invalid empty data"))
  157. }
  158. }
  159. func TestFSDelete(t *testing.T) {
  160. store, cleanup := defaultFSStoreBackend(t)
  161. defer cleanup()
  162. id, err := store.Set([]byte("foo"))
  163. assert.Check(t, err)
  164. id2, err := store.Set([]byte("bar"))
  165. assert.Check(t, err)
  166. err = store.Delete(id)
  167. assert.Check(t, err)
  168. _, err = store.Get(id)
  169. assert.Check(t, is.ErrorContains(err, "failed to get digest"))
  170. _, err = store.Get(id2)
  171. assert.Check(t, err)
  172. err = store.Delete(id2)
  173. assert.Check(t, err)
  174. _, err = store.Get(id2)
  175. assert.Check(t, is.ErrorContains(err, "failed to get digest"))
  176. }
  177. func TestFSWalker(t *testing.T) {
  178. store, cleanup := defaultFSStoreBackend(t)
  179. defer cleanup()
  180. id, err := store.Set([]byte("foo"))
  181. assert.Check(t, err)
  182. id2, err := store.Set([]byte("bar"))
  183. assert.Check(t, err)
  184. tcases := make(map[digest.Digest]struct{})
  185. tcases[id] = struct{}{}
  186. tcases[id2] = struct{}{}
  187. n := 0
  188. err = store.Walk(func(id digest.Digest) error {
  189. delete(tcases, id)
  190. n++
  191. return nil
  192. })
  193. assert.Check(t, err)
  194. assert.Check(t, is.Equal(2, n))
  195. assert.Check(t, is.Len(tcases, 0))
  196. }
  197. func TestFSWalkerStopOnError(t *testing.T) {
  198. store, cleanup := defaultFSStoreBackend(t)
  199. defer cleanup()
  200. id, err := store.Set([]byte("foo"))
  201. assert.Check(t, err)
  202. tcases := make(map[digest.Digest]struct{})
  203. tcases[id] = struct{}{}
  204. err = store.Walk(func(id digest.Digest) error {
  205. return errors.New("what")
  206. })
  207. assert.Check(t, is.ErrorContains(err, "what"))
  208. }