set.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. package digestset
  2. import (
  3. "errors"
  4. "sort"
  5. "strings"
  6. "sync"
  7. digest "github.com/opencontainers/go-digest"
  8. )
  9. var (
  10. // ErrDigestNotFound is used when a matching digest
  11. // could not be found in a set.
  12. ErrDigestNotFound = errors.New("digest not found")
  13. // ErrDigestAmbiguous is used when multiple digests
  14. // are found in a set. None of the matching digests
  15. // should be considered valid matches.
  16. ErrDigestAmbiguous = errors.New("ambiguous digest string")
  17. )
  18. // Set is used to hold a unique set of digests which
  19. // may be easily referenced by easily referenced by a string
  20. // representation of the digest as well as short representation.
  21. // The uniqueness of the short representation is based on other
  22. // digests in the set. If digests are omitted from this set,
  23. // collisions in a larger set may not be detected, therefore it
  24. // is important to always do short representation lookups on
  25. // the complete set of digests. To mitigate collisions, an
  26. // appropriately long short code should be used.
  27. type Set struct {
  28. mutex sync.RWMutex
  29. entries digestEntries
  30. }
  31. // NewSet creates an empty set of digests
  32. // which may have digests added.
  33. func NewSet() *Set {
  34. return &Set{
  35. entries: digestEntries{},
  36. }
  37. }
  38. // checkShortMatch checks whether two digests match as either whole
  39. // values or short values. This function does not test equality,
  40. // rather whether the second value could match against the first
  41. // value.
  42. func checkShortMatch(alg digest.Algorithm, hex, shortAlg, shortHex string) bool {
  43. if len(hex) == len(shortHex) {
  44. if hex != shortHex {
  45. return false
  46. }
  47. if len(shortAlg) > 0 && string(alg) != shortAlg {
  48. return false
  49. }
  50. } else if !strings.HasPrefix(hex, shortHex) {
  51. return false
  52. } else if len(shortAlg) > 0 && string(alg) != shortAlg {
  53. return false
  54. }
  55. return true
  56. }
  57. // Lookup looks for a digest matching the given string representation.
  58. // If no digests could be found ErrDigestNotFound will be returned
  59. // with an empty digest value. If multiple matches are found
  60. // ErrDigestAmbiguous will be returned with an empty digest value.
  61. func (dst *Set) Lookup(d string) (digest.Digest, error) {
  62. dst.mutex.RLock()
  63. defer dst.mutex.RUnlock()
  64. if len(dst.entries) == 0 {
  65. return "", ErrDigestNotFound
  66. }
  67. var (
  68. searchFunc func(int) bool
  69. alg digest.Algorithm
  70. hex string
  71. )
  72. dgst, err := digest.Parse(d)
  73. if err == digest.ErrDigestInvalidFormat {
  74. hex = d
  75. searchFunc = func(i int) bool {
  76. return dst.entries[i].val >= d
  77. }
  78. } else {
  79. hex = dgst.Hex()
  80. alg = dgst.Algorithm()
  81. searchFunc = func(i int) bool {
  82. if dst.entries[i].val == hex {
  83. return dst.entries[i].alg >= alg
  84. }
  85. return dst.entries[i].val >= hex
  86. }
  87. }
  88. idx := sort.Search(len(dst.entries), searchFunc)
  89. if idx == len(dst.entries) || !checkShortMatch(dst.entries[idx].alg, dst.entries[idx].val, string(alg), hex) {
  90. return "", ErrDigestNotFound
  91. }
  92. if dst.entries[idx].alg == alg && dst.entries[idx].val == hex {
  93. return dst.entries[idx].digest, nil
  94. }
  95. if idx+1 < len(dst.entries) && checkShortMatch(dst.entries[idx+1].alg, dst.entries[idx+1].val, string(alg), hex) {
  96. return "", ErrDigestAmbiguous
  97. }
  98. return dst.entries[idx].digest, nil
  99. }
  100. // Add adds the given digest to the set. An error will be returned
  101. // if the given digest is invalid. If the digest already exists in the
  102. // set, this operation will be a no-op.
  103. func (dst *Set) Add(d digest.Digest) error {
  104. if err := d.Validate(); err != nil {
  105. return err
  106. }
  107. dst.mutex.Lock()
  108. defer dst.mutex.Unlock()
  109. entry := &digestEntry{alg: d.Algorithm(), val: d.Hex(), digest: d}
  110. searchFunc := func(i int) bool {
  111. if dst.entries[i].val == entry.val {
  112. return dst.entries[i].alg >= entry.alg
  113. }
  114. return dst.entries[i].val >= entry.val
  115. }
  116. idx := sort.Search(len(dst.entries), searchFunc)
  117. if idx == len(dst.entries) {
  118. dst.entries = append(dst.entries, entry)
  119. return nil
  120. } else if dst.entries[idx].digest == d {
  121. return nil
  122. }
  123. entries := append(dst.entries, nil)
  124. copy(entries[idx+1:], entries[idx:len(entries)-1])
  125. entries[idx] = entry
  126. dst.entries = entries
  127. return nil
  128. }
  129. // Remove removes the given digest from the set. An err will be
  130. // returned if the given digest is invalid. If the digest does
  131. // not exist in the set, this operation will be a no-op.
  132. func (dst *Set) Remove(d digest.Digest) error {
  133. if err := d.Validate(); err != nil {
  134. return err
  135. }
  136. dst.mutex.Lock()
  137. defer dst.mutex.Unlock()
  138. entry := &digestEntry{alg: d.Algorithm(), val: d.Hex(), digest: d}
  139. searchFunc := func(i int) bool {
  140. if dst.entries[i].val == entry.val {
  141. return dst.entries[i].alg >= entry.alg
  142. }
  143. return dst.entries[i].val >= entry.val
  144. }
  145. idx := sort.Search(len(dst.entries), searchFunc)
  146. // Not found if idx is after or value at idx is not digest
  147. if idx == len(dst.entries) || dst.entries[idx].digest != d {
  148. return nil
  149. }
  150. entries := dst.entries
  151. copy(entries[idx:], entries[idx+1:])
  152. entries = entries[:len(entries)-1]
  153. dst.entries = entries
  154. return nil
  155. }
  156. // All returns all the digests in the set
  157. func (dst *Set) All() []digest.Digest {
  158. dst.mutex.RLock()
  159. defer dst.mutex.RUnlock()
  160. retValues := make([]digest.Digest, len(dst.entries))
  161. for i := range dst.entries {
  162. retValues[i] = dst.entries[i].digest
  163. }
  164. return retValues
  165. }
  166. // ShortCodeTable returns a map of Digest to unique short codes. The
  167. // length represents the minimum value, the maximum length may be the
  168. // entire value of digest if uniqueness cannot be achieved without the
  169. // full value. This function will attempt to make short codes as short
  170. // as possible to be unique.
  171. func ShortCodeTable(dst *Set, length int) map[digest.Digest]string {
  172. dst.mutex.RLock()
  173. defer dst.mutex.RUnlock()
  174. m := make(map[digest.Digest]string, len(dst.entries))
  175. l := length
  176. resetIdx := 0
  177. for i := 0; i < len(dst.entries); i++ {
  178. var short string
  179. extended := true
  180. for extended {
  181. extended = false
  182. if len(dst.entries[i].val) <= l {
  183. short = dst.entries[i].digest.String()
  184. } else {
  185. short = dst.entries[i].val[:l]
  186. for j := i + 1; j < len(dst.entries); j++ {
  187. if checkShortMatch(dst.entries[j].alg, dst.entries[j].val, "", short) {
  188. if j > resetIdx {
  189. resetIdx = j
  190. }
  191. extended = true
  192. } else {
  193. break
  194. }
  195. }
  196. if extended {
  197. l++
  198. }
  199. }
  200. }
  201. m[dst.entries[i].digest] = short
  202. if i >= resetIdx {
  203. l = length
  204. }
  205. }
  206. return m
  207. }
  208. type digestEntry struct {
  209. alg digest.Algorithm
  210. val string
  211. digest digest.Digest
  212. }
  213. type digestEntries []*digestEntry
  214. func (d digestEntries) Len() int {
  215. return len(d)
  216. }
  217. func (d digestEntries) Less(i, j int) bool {
  218. if d[i].val != d[j].val {
  219. return d[i].val < d[j].val
  220. }
  221. return d[i].alg < d[j].alg
  222. }
  223. func (d digestEntries) Swap(i, j int) {
  224. d[i], d[j] = d[j], d[i]
  225. }