ref_test.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. package blob
  2. import (
  3. "strings"
  4. "testing"
  5. )
  6. var testRefs = map[string]Ref{
  7. "mistral:latest": {name: "mistral", tag: "latest"},
  8. "mistral": {name: "mistral"},
  9. "mistral:30B": {name: "mistral", tag: "30B"},
  10. "mistral:7b": {name: "mistral", tag: "7b"},
  11. "mistral:7b+Q4_0": {name: "mistral", tag: "7b", build: "Q4_0"},
  12. "mistral+KQED": {name: "mistral", build: "KQED"},
  13. "mistral.x-3:7b+Q4_0": {name: "mistral.x-3", tag: "7b", build: "Q4_0"},
  14. "mistral:7b+q4_0": {name: "mistral", tag: "7b", build: "Q4_0"},
  15. "llama2": {name: "llama2"},
  16. // invalid (includes fuzzing trophies)
  17. "+": {},
  18. "mistral:7b+Q4_0:latest": {},
  19. "mi tral": {},
  20. "x/y/z/foo": {},
  21. "/0": {},
  22. "0 /0": {},
  23. "0 /": {},
  24. "0/": {},
  25. ":": {},
  26. ":/0": {},
  27. "+0/00000": {},
  28. "0+.\xf2\x80\xf6\x9d00000\xe5\x99\xe6\xd900\xd90\xa60\x91\xdc0\xff\xbf\x99\xe800\xb9\xdc\xd6\xc300\x970\xfb\xfd0\xe0\x8a\xe1\xad\xd40\x9700\xa80\x980\xdd0000\xb00\x91000\xfe0\x89\x9b\x90\x93\x9f0\xe60\xf7\x84\xb0\x87\xa5\xff0\xa000\x9a\x85\xf6\x85\xfe\xa9\xf9\xe9\xde00\xf4\xe0\x8f\x81\xad\xde00\xd700\xaa\xe000000\xb1\xee0\x91": {},
  29. "0//0": {},
  30. "m+^^^": {},
  31. "file:///etc/passwd": {},
  32. "file:///etc/passwd:latest": {},
  33. "file:///etc/passwd:latest+u": {},
  34. strings.Repeat("a", MaxRefLength): {name: strings.Repeat("a", MaxRefLength)},
  35. strings.Repeat("a", MaxRefLength+1): {},
  36. }
  37. func TestRefParts(t *testing.T) {
  38. const wantNumParts = 5
  39. var ref Ref
  40. if len(ref.Parts()) != wantNumParts {
  41. t.Errorf("Parts() = %d; want %d", len(ref.Parts()), wantNumParts)
  42. }
  43. }
  44. func TestParseRef(t *testing.T) {
  45. for s, want := range testRefs {
  46. for _, prefix := range []string{"", "https://", "http://"} {
  47. // We should get the same results with or without the
  48. // http(s) prefixes
  49. s := prefix + s
  50. t.Run(s, func(t *testing.T) {
  51. got := ParseRef(s)
  52. if got != want {
  53. t.Errorf("ParseRef(%q) = %q; want %q", s, got, want)
  54. }
  55. // test round-trip
  56. if ParseRef(got.String()) != got {
  57. t.Errorf("String() = %s; want %s", got.String(), s)
  58. }
  59. })
  60. }
  61. }
  62. }
  63. func TestRefComplete(t *testing.T) {
  64. cases := []struct {
  65. in string
  66. complete bool
  67. completeWithoutBuild bool
  68. }{
  69. {"", false, false},
  70. {"example.com/mistral:7b+x", false, false},
  71. {"example.com/mistral:7b+Q4_0", false, false},
  72. {"mistral:7b+x", false, false},
  73. {"example.com/x/mistral:latest+Q4_0", true, true},
  74. {"example.com/x/mistral:latest", false, true},
  75. }
  76. for _, tt := range cases {
  77. t.Run(tt.in, func(t *testing.T) {
  78. ref := ParseRef(tt.in)
  79. t.Logf("ParseRef(%q) = %#v", tt.in, ref)
  80. if g := ref.Complete(); g != tt.complete {
  81. t.Errorf("Complete(%q) = %v; want %v", tt.in, g, tt.complete)
  82. }
  83. if g := ref.CompleteWithoutBuild(); g != tt.completeWithoutBuild {
  84. t.Errorf("CompleteWithoutBuild(%q) = %v; want %v", tt.in, g, tt.completeWithoutBuild)
  85. }
  86. })
  87. }
  88. }
  89. func TestRefStringVariants(t *testing.T) {
  90. cases := []struct {
  91. in string
  92. nameAndTag string
  93. nameTagAndBuild string
  94. }{
  95. {"x/y/z:8n+I", "z:8n", "z:8n+I"},
  96. {"x/y/z:8n", "z:8n", "z:8n"},
  97. }
  98. for _, tt := range cases {
  99. t.Run(tt.in, func(t *testing.T) {
  100. ref := ParseRef(tt.in)
  101. t.Logf("ParseRef(%q) = %#v", tt.in, ref)
  102. if g := ref.NameAndTag(); g != tt.nameAndTag {
  103. t.Errorf("NameAndTag(%q) = %q; want %q", tt.in, g, tt.nameAndTag)
  104. }
  105. if g := ref.NameTagAndBuild(); g != tt.nameTagAndBuild {
  106. t.Errorf("NameTagAndBuild(%q) = %q; want %q", tt.in, g, tt.nameTagAndBuild)
  107. }
  108. })
  109. }
  110. }
  111. func TestRefFull(t *testing.T) {
  112. const empty = "!(MISSING DOMAIN)/!(MISSING NAMESPACE)/!(MISSING NAME):!(MISSING TAG)+!(MISSING BUILD)"
  113. cases := []struct {
  114. in string
  115. wantFull string
  116. }{
  117. {"", empty},
  118. {"example.com/mistral:7b+x", "!(MISSING DOMAIN)/example.com/mistral:7b+X"},
  119. {"example.com/mistral:7b+Q4_0", "!(MISSING DOMAIN)/example.com/mistral:7b+Q4_0"},
  120. {"example.com/x/mistral:latest", "example.com/x/mistral:latest+!(MISSING BUILD)"},
  121. {"example.com/x/mistral:latest+Q4_0", "example.com/x/mistral:latest+Q4_0"},
  122. {"mistral:7b+x", "!(MISSING DOMAIN)/!(MISSING NAMESPACE)/mistral:7b+X"},
  123. {"mistral:7b+q4_0", "!(MISSING DOMAIN)/!(MISSING NAMESPACE)/mistral:7b+Q4_0"},
  124. {"mistral:7b+Q4_0", "!(MISSING DOMAIN)/!(MISSING NAMESPACE)/mistral:7b+Q4_0"},
  125. {"mistral:latest", "!(MISSING DOMAIN)/!(MISSING NAMESPACE)/mistral:latest+!(MISSING BUILD)"},
  126. {"mistral", "!(MISSING DOMAIN)/!(MISSING NAMESPACE)/mistral:!(MISSING TAG)+!(MISSING BUILD)"},
  127. {"mistral:30b", "!(MISSING DOMAIN)/!(MISSING NAMESPACE)/mistral:30b+!(MISSING BUILD)"},
  128. }
  129. for _, tt := range cases {
  130. t.Run(tt.in, func(t *testing.T) {
  131. ref := ParseRef(tt.in)
  132. t.Logf("ParseRef(%q) = %#v", tt.in, ref)
  133. if g := ref.Full(); g != tt.wantFull {
  134. t.Errorf("Full(%q) = %q; want %q", tt.in, g, tt.wantFull)
  135. }
  136. })
  137. }
  138. }
  139. func TestParseRefAllocs(t *testing.T) {
  140. // test allocations
  141. var r Ref
  142. allocs := testing.AllocsPerRun(1000, func() {
  143. r = ParseRef("example.com/mistral:7b+Q4_0")
  144. })
  145. _ = r
  146. if allocs > 0 {
  147. t.Errorf("ParseRef allocs = %v; want 0", allocs)
  148. }
  149. }
  150. func BenchmarkParseRef(b *testing.B) {
  151. b.ReportAllocs()
  152. var r Ref
  153. for i := 0; i < b.N; i++ {
  154. r = ParseRef("example.com/mistral:7b+Q4_0")
  155. }
  156. _ = r
  157. }
  158. func FuzzParseRef(f *testing.F) {
  159. f.Add("example.com/mistral:7b+Q4_0")
  160. f.Add("example.com/mistral:7b+q4_0")
  161. f.Add("example.com/mistral:7b+x")
  162. f.Add("x/y/z:8n+I")
  163. f.Fuzz(func(t *testing.T, s string) {
  164. r0 := ParseRef(s)
  165. if !r0.Valid() {
  166. if r0 != (Ref{}) {
  167. t.Errorf("expected invalid ref to be zero value; got %#v", r0)
  168. }
  169. t.Skipf("invalid ref: %q", s)
  170. }
  171. for _, p := range r0.Parts() {
  172. if len(p) > MaxRefLength {
  173. t.Errorf("part too long: %q", p)
  174. }
  175. }
  176. if !strings.EqualFold(r0.String(), s) {
  177. t.Errorf("String() did not round-trip with case insensitivity: %q\ngot = %q\nwant = %q", s, r0.String(), s)
  178. }
  179. r1 := ParseRef(r0.String())
  180. if r0 != r1 {
  181. t.Errorf("round-trip mismatch: %+v != %+v", r0, r1)
  182. }
  183. })
  184. }