/util/semver/semver.go

https://github.com/robert-zaremba/nsq · Go · 173 lines · 128 code · 14 blank · 31 comment · 36 complexity · 615751737dcd6cb4ddd909e29bb2cc65 MD5 · raw file

  1. // COPIED FROM http://code.google.com/p/go-semver/
  2. //
  3. // The version package implements semantic versions as described
  4. // at http://semver.org/
  5. package semver
  6. import (
  7. "fmt"
  8. "regexp"
  9. "strconv"
  10. "strings"
  11. )
  12. // Version represents a parsed version. See http://semver.org/ for
  13. // detailed description of the various components.
  14. type Version struct {
  15. Major int // The major version number.
  16. Minor int // The minor version number.
  17. Patch int // The patch version number.
  18. Prerelease []string // The pre-release version (dot-separated elements)
  19. Build []string // The build version (dot-separated elements)
  20. }
  21. var charClasses = strings.NewReplacer("d", `[0-9]`, "c", `[\-0-9a-z]`)
  22. const pattern = `^(d{1,9})\.(d{1,9})\.(d{1,9})(-c+(\.c+)*)?(\+c+(\.c+)*)?$`
  23. var versionPat = regexp.MustCompile(charClasses.Replace(pattern))
  24. // Parse parses the version, which is of one of the following forms:
  25. // 1.2.3
  26. // 1.2.3-prerelease
  27. // 1.2.3+build
  28. // 1.2.3-prerelease+build
  29. func Parse(s string) (*Version, error) {
  30. m := versionPat.FindStringSubmatch(s)
  31. if m == nil {
  32. return nil, fmt.Errorf("invalid version %q", s)
  33. }
  34. v := new(Version)
  35. v.Major = atoi(m[1])
  36. v.Minor = atoi(m[2])
  37. v.Patch = atoi(m[3])
  38. if m[4] != "" {
  39. v.Prerelease = strings.Split(m[4][1:], ".")
  40. }
  41. if m[6] != "" {
  42. v.Build = strings.Split(m[6][1:], ".")
  43. }
  44. return v, nil
  45. }
  46. // atoi is the same as strconv.Atoi but assumes that
  47. // the string has been verified to be a valid integer.
  48. func atoi(s string) int {
  49. n, err := strconv.Atoi(s)
  50. if err != nil {
  51. panic(err)
  52. }
  53. return n
  54. }
  55. func (v *Version) String() string {
  56. var pre, build string
  57. if v.Prerelease != nil {
  58. pre = "-" + strings.Join(v.Prerelease, ".")
  59. }
  60. if v.Build != nil {
  61. build = "+" + strings.Join(v.Build, ".")
  62. }
  63. return fmt.Sprintf("%d.%d.%d%s%s", v.Major, v.Minor, v.Patch, pre, build)
  64. }
  65. func allDigits(s string) bool {
  66. for _, c := range s {
  67. if c < '0' || c > '9' {
  68. return false
  69. }
  70. }
  71. return true
  72. }
  73. // lessIds returns whether the slice of identifiers a is less than b,
  74. // as specified in semver.org,
  75. func lessIds(a, b []string) (v bool) {
  76. i := 0
  77. for ; i < len(a) && i < len(b); i++ {
  78. if c := cmp(a[i], b[i]); c != 0 {
  79. return c < 0
  80. }
  81. }
  82. return i < len(b)
  83. }
  84. // eqIds returns whether the slice of identifiers a is equal to b,
  85. // as specified in semver.org,
  86. func eqIds(a, b []string) bool {
  87. if len(a) != len(b) {
  88. return false
  89. }
  90. for i, s := range a {
  91. if cmp(s, b[i]) != 0 {
  92. return false
  93. }
  94. }
  95. return true
  96. }
  97. // cmp implements comparison of identifiers as specified at semver.org.
  98. // It returns 1, -1 or 0 if a is greater-than, less-than or equal to b,
  99. // respectively.
  100. //
  101. // Identifiers consisting of only digits are compared numerically and
  102. // identifiers with letters or dashes are compared lexically in ASCII
  103. // sort order. Numeric identifiers always have lower precedence than
  104. // non-numeric identifiers.
  105. func cmp(a, b string) int {
  106. numa, numb := allDigits(a), allDigits(b)
  107. switch {
  108. case numa && numb:
  109. return numCmp(a, b)
  110. case numa:
  111. return -1
  112. case numb:
  113. return 1
  114. case a < b:
  115. return -1
  116. case a > b:
  117. return 1
  118. }
  119. return 0
  120. }
  121. // numCmp 1, -1 or 0 depending on whether the known-to-be-all-digits
  122. // strings a and b are numerically greater than, less than or equal to
  123. // each other. Avoiding the conversion means we can work correctly with
  124. // very long version numbers.
  125. func numCmp(a, b string) int {
  126. a = strings.TrimLeft(a, "0")
  127. b = strings.TrimLeft(b, "0")
  128. switch {
  129. case len(a) < len(b):
  130. return -1
  131. case len(a) > len(b):
  132. return 1
  133. case a < b:
  134. return -1
  135. case a > b:
  136. return 1
  137. }
  138. return 0
  139. }
  140. // Less returns whether v is semantically earlier in the
  141. // version sequence than w.
  142. func (v *Version) Less(w *Version) bool {
  143. switch {
  144. case v.Major != w.Major:
  145. return v.Major < w.Major
  146. case v.Minor != w.Minor:
  147. return v.Minor < w.Minor
  148. case v.Patch != w.Patch:
  149. return v.Patch < w.Patch
  150. case !eqIds(v.Prerelease, w.Prerelease):
  151. if v.Prerelease == nil || w.Prerelease == nil {
  152. return v.Prerelease != nil
  153. }
  154. return lessIds(v.Prerelease, w.Prerelease)
  155. case !eqIds(v.Build, w.Build):
  156. return lessIds(v.Build, w.Build)
  157. }
  158. return false
  159. }