store_test.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. package reference // import "github.com/docker/docker/reference"
  2. import (
  3. "bytes"
  4. "io/ioutil"
  5. "os"
  6. "path/filepath"
  7. "strings"
  8. "testing"
  9. "github.com/docker/distribution/reference"
  10. digest "github.com/opencontainers/go-digest"
  11. "gotest.tools/v3/assert"
  12. is "gotest.tools/v3/assert/cmp"
  13. )
  14. var (
  15. saveLoadTestCases = map[string]digest.Digest{
  16. "registry:5000/foobar:HEAD": "sha256:470022b8af682154f57a2163d030eb369549549cba00edc69e1b99b46bb924d6",
  17. "registry:5000/foobar:alternate": "sha256:ae300ebc4a4f00693702cfb0a5e0b7bc527b353828dc86ad09fb95c8a681b793",
  18. "registry:5000/foobar:latest": "sha256:6153498b9ac00968d71b66cca4eac37e990b5f9eb50c26877eb8799c8847451b",
  19. "registry:5000/foobar:master": "sha256:6c9917af4c4e05001b346421959d7ea81b6dc9d25718466a37a6add865dfd7fc",
  20. "jess/hollywood:latest": "sha256:ae7a5519a0a55a2d4ef20ddcbd5d0ca0888a1f7ab806acc8e2a27baf46f529fe",
  21. "registry@sha256:367eb40fd0330a7e464777121e39d2f5b3e8e23a1e159342e53ab05c9e4d94e6": "sha256:24126a56805beb9711be5f4590cc2eb55ab8d4a85ebd618eed72bb19fc50631c",
  22. "busybox:latest": "sha256:91e54dfb11794fad694460162bf0cb0a4fa710cfa3f60979c177d920813e267c",
  23. }
  24. marshalledSaveLoadTestCases = []byte(`{"Repositories":{"busybox":{"busybox:latest":"sha256:91e54dfb11794fad694460162bf0cb0a4fa710cfa3f60979c177d920813e267c"},"jess/hollywood":{"jess/hollywood:latest":"sha256:ae7a5519a0a55a2d4ef20ddcbd5d0ca0888a1f7ab806acc8e2a27baf46f529fe"},"registry":{"registry@sha256:367eb40fd0330a7e464777121e39d2f5b3e8e23a1e159342e53ab05c9e4d94e6":"sha256:24126a56805beb9711be5f4590cc2eb55ab8d4a85ebd618eed72bb19fc50631c"},"registry:5000/foobar":{"registry:5000/foobar:HEAD":"sha256:470022b8af682154f57a2163d030eb369549549cba00edc69e1b99b46bb924d6","registry:5000/foobar:alternate":"sha256:ae300ebc4a4f00693702cfb0a5e0b7bc527b353828dc86ad09fb95c8a681b793","registry:5000/foobar:latest":"sha256:6153498b9ac00968d71b66cca4eac37e990b5f9eb50c26877eb8799c8847451b","registry:5000/foobar:master":"sha256:6c9917af4c4e05001b346421959d7ea81b6dc9d25718466a37a6add865dfd7fc"}}}`)
  25. )
  26. func TestLoad(t *testing.T) {
  27. jsonFile, err := ioutil.TempFile("", "tag-store-test")
  28. if err != nil {
  29. t.Fatalf("error creating temp file: %v", err)
  30. }
  31. defer os.RemoveAll(jsonFile.Name())
  32. // Write canned json to the temp file
  33. _, err = jsonFile.Write(marshalledSaveLoadTestCases)
  34. if err != nil {
  35. t.Fatalf("error writing to temp file: %v", err)
  36. }
  37. jsonFile.Close()
  38. store, err := NewReferenceStore(jsonFile.Name())
  39. if err != nil {
  40. t.Fatalf("error creating tag store: %v", err)
  41. }
  42. for refStr, expectedID := range saveLoadTestCases {
  43. ref, err := reference.ParseNormalizedNamed(refStr)
  44. if err != nil {
  45. t.Fatalf("failed to parse reference: %v", err)
  46. }
  47. id, err := store.Get(ref)
  48. if err != nil {
  49. t.Fatalf("could not find reference %s: %v", refStr, err)
  50. }
  51. if id != expectedID {
  52. t.Fatalf("expected %s - got %s", expectedID, id)
  53. }
  54. }
  55. }
  56. func TestSave(t *testing.T) {
  57. jsonFile, err := ioutil.TempFile("", "tag-store-test")
  58. assert.NilError(t, err)
  59. _, err = jsonFile.Write([]byte(`{}`))
  60. assert.NilError(t, err)
  61. jsonFile.Close()
  62. defer os.RemoveAll(jsonFile.Name())
  63. store, err := NewReferenceStore(jsonFile.Name())
  64. if err != nil {
  65. t.Fatalf("error creating tag store: %v", err)
  66. }
  67. for refStr, id := range saveLoadTestCases {
  68. ref, err := reference.ParseNormalizedNamed(refStr)
  69. if err != nil {
  70. t.Fatalf("failed to parse reference: %v", err)
  71. }
  72. if canonical, ok := ref.(reference.Canonical); ok {
  73. err = store.AddDigest(canonical, id, false)
  74. if err != nil {
  75. t.Fatalf("could not add digest reference %s: %v", refStr, err)
  76. }
  77. } else {
  78. err = store.AddTag(ref, id, false)
  79. if err != nil {
  80. t.Fatalf("could not add reference %s: %v", refStr, err)
  81. }
  82. }
  83. }
  84. jsonBytes, err := ioutil.ReadFile(jsonFile.Name())
  85. if err != nil {
  86. t.Fatalf("could not read json file: %v", err)
  87. }
  88. if !bytes.Equal(jsonBytes, marshalledSaveLoadTestCases) {
  89. t.Fatalf("save output did not match expectations\nexpected:\n%s\ngot:\n%s", marshalledSaveLoadTestCases, jsonBytes)
  90. }
  91. }
  92. func TestAddDeleteGet(t *testing.T) {
  93. jsonFile, err := ioutil.TempFile("", "tag-store-test")
  94. if err != nil {
  95. t.Fatalf("error creating temp file: %v", err)
  96. }
  97. _, err = jsonFile.Write([]byte(`{}`))
  98. assert.NilError(t, err)
  99. _ = jsonFile.Close()
  100. defer func() { _ = os.RemoveAll(jsonFile.Name()) }()
  101. store, err := NewReferenceStore(jsonFile.Name())
  102. if err != nil {
  103. t.Fatalf("error creating tag store: %v", err)
  104. }
  105. testImageID1 := digest.Digest("sha256:9655aef5fd742a1b4e1b7b163aa9f1c76c186304bf39102283d80927c916ca9c")
  106. testImageID2 := digest.Digest("sha256:9655aef5fd742a1b4e1b7b163aa9f1c76c186304bf39102283d80927c916ca9d")
  107. testImageID3 := digest.Digest("sha256:9655aef5fd742a1b4e1b7b163aa9f1c76c186304bf39102283d80927c916ca9e")
  108. // Try adding a reference with no tag or digest
  109. nameOnly, err := reference.ParseNormalizedNamed("username/repo")
  110. if err != nil {
  111. t.Fatalf("could not parse reference: %v", err)
  112. }
  113. if err = store.AddTag(nameOnly, testImageID1, false); err != nil {
  114. t.Fatalf("error adding to store: %v", err)
  115. }
  116. // Add a few references
  117. ref1, err := reference.ParseNormalizedNamed("username/repo1:latest")
  118. if err != nil {
  119. t.Fatalf("could not parse reference: %v", err)
  120. }
  121. if err = store.AddTag(ref1, testImageID1, false); err != nil {
  122. t.Fatalf("error adding to store: %v", err)
  123. }
  124. ref2, err := reference.ParseNormalizedNamed("username/repo1:old")
  125. if err != nil {
  126. t.Fatalf("could not parse reference: %v", err)
  127. }
  128. if err = store.AddTag(ref2, testImageID2, false); err != nil {
  129. t.Fatalf("error adding to store: %v", err)
  130. }
  131. ref3, err := reference.ParseNormalizedNamed("username/repo1:alias")
  132. if err != nil {
  133. t.Fatalf("could not parse reference: %v", err)
  134. }
  135. if err = store.AddTag(ref3, testImageID1, false); err != nil {
  136. t.Fatalf("error adding to store: %v", err)
  137. }
  138. ref4, err := reference.ParseNormalizedNamed("username/repo2:latest")
  139. if err != nil {
  140. t.Fatalf("could not parse reference: %v", err)
  141. }
  142. if err = store.AddTag(ref4, testImageID2, false); err != nil {
  143. t.Fatalf("error adding to store: %v", err)
  144. }
  145. // Write the same values again; should silently succeed
  146. if err = store.AddTag(ref4, testImageID2, false); err != nil {
  147. t.Fatalf("error redundantly adding to store: %v", err)
  148. }
  149. ref5, err := reference.ParseNormalizedNamed("username/repo3@sha256:58153dfb11794fad694460162bf0cb0a4fa710cfa3f60979c177d920813e267c")
  150. if err != nil {
  151. t.Fatalf("could not parse reference: %v", err)
  152. }
  153. if err = store.AddDigest(ref5.(reference.Canonical), testImageID2, false); err != nil {
  154. t.Fatalf("error adding to store: %v", err)
  155. }
  156. // Write the same values again; should silently succeed
  157. if err = store.AddDigest(ref5.(reference.Canonical), testImageID2, false); err != nil {
  158. t.Fatalf("error redundantly adding to store: %v", err)
  159. }
  160. // Attempt to overwrite with force == false
  161. if err = store.AddTag(ref4, testImageID3, false); err == nil || !strings.HasPrefix(err.Error(), "Conflict:") {
  162. t.Fatalf("did not get expected error on overwrite attempt - got %v", err)
  163. }
  164. // Repeat to overwrite with force == true
  165. if err = store.AddTag(ref4, testImageID3, true); err != nil {
  166. t.Fatalf("failed to force tag overwrite: %v", err)
  167. }
  168. // Check references so far
  169. id, err := store.Get(nameOnly)
  170. if err != nil {
  171. t.Fatalf("Get returned error: %v", err)
  172. }
  173. if id != testImageID1 {
  174. t.Fatalf("id mismatch: got %s instead of %s", id.String(), testImageID1.String())
  175. }
  176. id, err = store.Get(ref1)
  177. if err != nil {
  178. t.Fatalf("Get returned error: %v", err)
  179. }
  180. if id != testImageID1 {
  181. t.Fatalf("id mismatch: got %s instead of %s", id.String(), testImageID1.String())
  182. }
  183. id, err = store.Get(ref2)
  184. if err != nil {
  185. t.Fatalf("Get returned error: %v", err)
  186. }
  187. if id != testImageID2 {
  188. t.Fatalf("id mismatch: got %s instead of %s", id.String(), testImageID2.String())
  189. }
  190. id, err = store.Get(ref3)
  191. if err != nil {
  192. t.Fatalf("Get returned error: %v", err)
  193. }
  194. if id != testImageID1 {
  195. t.Fatalf("id mismatch: got %s instead of %s", id.String(), testImageID1.String())
  196. }
  197. id, err = store.Get(ref4)
  198. if err != nil {
  199. t.Fatalf("Get returned error: %v", err)
  200. }
  201. if id != testImageID3 {
  202. t.Fatalf("id mismatch: got %s instead of %s", id.String(), testImageID3.String())
  203. }
  204. id, err = store.Get(ref5)
  205. if err != nil {
  206. t.Fatalf("Get returned error: %v", err)
  207. }
  208. if id != testImageID2 {
  209. t.Fatalf("id mismatch: got %s instead of %s", id.String(), testImageID3.String())
  210. }
  211. // Get should return ErrDoesNotExist for a nonexistent repo
  212. nonExistRepo, err := reference.ParseNormalizedNamed("username/nonexistrepo:latest")
  213. if err != nil {
  214. t.Fatalf("could not parse reference: %v", err)
  215. }
  216. if _, err = store.Get(nonExistRepo); err != ErrDoesNotExist {
  217. t.Fatal("Expected ErrDoesNotExist from Get")
  218. }
  219. // Get should return ErrDoesNotExist for a nonexistent tag
  220. nonExistTag, err := reference.ParseNormalizedNamed("username/repo1:nonexist")
  221. if err != nil {
  222. t.Fatalf("could not parse reference: %v", err)
  223. }
  224. if _, err = store.Get(nonExistTag); err != ErrDoesNotExist {
  225. t.Fatal("Expected ErrDoesNotExist from Get")
  226. }
  227. // Check References
  228. refs := store.References(testImageID1)
  229. if len(refs) != 3 {
  230. t.Fatal("unexpected number of references")
  231. }
  232. // Looking for the references in this order verifies that they are
  233. // returned lexically sorted.
  234. if refs[0].String() != ref3.String() {
  235. t.Fatalf("unexpected reference: %v", refs[0].String())
  236. }
  237. if refs[1].String() != ref1.String() {
  238. t.Fatalf("unexpected reference: %v", refs[1].String())
  239. }
  240. if refs[2].String() != nameOnly.String()+":latest" {
  241. t.Fatalf("unexpected reference: %v", refs[2].String())
  242. }
  243. // Check ReferencesByName
  244. repoName, err := reference.ParseNormalizedNamed("username/repo1")
  245. if err != nil {
  246. t.Fatalf("could not parse reference: %v", err)
  247. }
  248. associations := store.ReferencesByName(repoName)
  249. if len(associations) != 3 {
  250. t.Fatal("unexpected number of associations")
  251. }
  252. // Looking for the associations in this order verifies that they are
  253. // returned lexically sorted.
  254. if associations[0].Ref.String() != ref3.String() {
  255. t.Fatalf("unexpected reference: %v", associations[0].Ref.String())
  256. }
  257. if associations[0].ID != testImageID1 {
  258. t.Fatalf("unexpected reference: %v", associations[0].Ref.String())
  259. }
  260. if associations[1].Ref.String() != ref1.String() {
  261. t.Fatalf("unexpected reference: %v", associations[1].Ref.String())
  262. }
  263. if associations[1].ID != testImageID1 {
  264. t.Fatalf("unexpected reference: %v", associations[1].Ref.String())
  265. }
  266. if associations[2].Ref.String() != ref2.String() {
  267. t.Fatalf("unexpected reference: %v", associations[2].Ref.String())
  268. }
  269. if associations[2].ID != testImageID2 {
  270. t.Fatalf("unexpected reference: %v", associations[2].Ref.String())
  271. }
  272. // Delete should return ErrDoesNotExist for a nonexistent repo
  273. if _, err = store.Delete(nonExistRepo); err != ErrDoesNotExist {
  274. t.Fatal("Expected ErrDoesNotExist from Delete")
  275. }
  276. // Delete should return ErrDoesNotExist for a nonexistent tag
  277. if _, err = store.Delete(nonExistTag); err != ErrDoesNotExist {
  278. t.Fatal("Expected ErrDoesNotExist from Delete")
  279. }
  280. // Delete a few references
  281. if deleted, err := store.Delete(ref1); err != nil || !deleted {
  282. t.Fatal("Delete failed")
  283. }
  284. if _, err := store.Get(ref1); err != ErrDoesNotExist {
  285. t.Fatal("Expected ErrDoesNotExist from Get")
  286. }
  287. if deleted, err := store.Delete(ref5); err != nil || !deleted {
  288. t.Fatal("Delete failed")
  289. }
  290. if _, err := store.Get(ref5); err != ErrDoesNotExist {
  291. t.Fatal("Expected ErrDoesNotExist from Get")
  292. }
  293. if deleted, err := store.Delete(nameOnly); err != nil || !deleted {
  294. t.Fatal("Delete failed")
  295. }
  296. if _, err := store.Get(nameOnly); err != ErrDoesNotExist {
  297. t.Fatal("Expected ErrDoesNotExist from Get")
  298. }
  299. }
  300. func TestInvalidTags(t *testing.T) {
  301. tmpDir, err := ioutil.TempDir("", "tag-store-test")
  302. assert.NilError(t, err)
  303. defer os.RemoveAll(tmpDir)
  304. store, err := NewReferenceStore(filepath.Join(tmpDir, "repositories.json"))
  305. assert.NilError(t, err)
  306. id := digest.Digest("sha256:470022b8af682154f57a2163d030eb369549549cba00edc69e1b99b46bb924d6")
  307. // sha256 as repo name
  308. ref, err := reference.ParseNormalizedNamed("sha256:abc")
  309. assert.NilError(t, err)
  310. err = store.AddTag(ref, id, true)
  311. assert.Check(t, is.ErrorContains(err, ""))
  312. // setting digest as a tag
  313. ref, err = reference.ParseNormalizedNamed("registry@sha256:367eb40fd0330a7e464777121e39d2f5b3e8e23a1e159342e53ab05c9e4d94e6")
  314. assert.NilError(t, err)
  315. err = store.AddTag(ref, id, true)
  316. assert.Check(t, is.ErrorContains(err, ""))
  317. }