/vendor/github.com/Masterminds/semver/version.go

https://bitbucket.org/butbox/traefik · Go · 375 lines · 247 code · 52 blank · 76 comment · 65 complexity · be98b003a8d80553d3058c9aba8b8e36 MD5 · raw file

  1. package semver
  2. import (
  3. "bytes"
  4. "errors"
  5. "fmt"
  6. "regexp"
  7. "strconv"
  8. "strings"
  9. )
  10. // The compiled version of the regex created at init() is cached here so it
  11. // only needs to be created once.
  12. var versionRegex *regexp.Regexp
  13. var validPrereleaseRegex *regexp.Regexp
  14. var (
  15. // ErrInvalidSemVer is returned a version is found to be invalid when
  16. // being parsed.
  17. ErrInvalidSemVer = errors.New("Invalid Semantic Version")
  18. // ErrInvalidMetadata is returned when the metadata is an invalid format
  19. ErrInvalidMetadata = errors.New("Invalid Metadata string")
  20. // ErrInvalidPrerelease is returned when the pre-release is an invalid format
  21. ErrInvalidPrerelease = errors.New("Invalid Prerelease string")
  22. )
  23. // SemVerRegex is the regular expression used to parse a semantic version.
  24. const SemVerRegex string = `v?([0-9]+)(\.[0-9]+)?(\.[0-9]+)?` +
  25. `(-([0-9A-Za-z\-]+(\.[0-9A-Za-z\-]+)*))?` +
  26. `(\+([0-9A-Za-z\-]+(\.[0-9A-Za-z\-]+)*))?`
  27. // ValidPrerelease is the regular expression which validates
  28. // both prerelease and metadata values.
  29. const ValidPrerelease string = `^([0-9A-Za-z\-]+(\.[0-9A-Za-z\-]+)*)`
  30. // Version represents a single semantic version.
  31. type Version struct {
  32. major, minor, patch int64
  33. pre string
  34. metadata string
  35. original string
  36. }
  37. func init() {
  38. versionRegex = regexp.MustCompile("^" + SemVerRegex + "$")
  39. validPrereleaseRegex = regexp.MustCompile(ValidPrerelease)
  40. }
  41. // NewVersion parses a given version and returns an instance of Version or
  42. // an error if unable to parse the version.
  43. func NewVersion(v string) (*Version, error) {
  44. m := versionRegex.FindStringSubmatch(v)
  45. if m == nil {
  46. return nil, ErrInvalidSemVer
  47. }
  48. sv := &Version{
  49. metadata: m[8],
  50. pre: m[5],
  51. original: v,
  52. }
  53. var temp int64
  54. temp, err := strconv.ParseInt(m[1], 10, 32)
  55. if err != nil {
  56. return nil, fmt.Errorf("Error parsing version segment: %s", err)
  57. }
  58. sv.major = temp
  59. if m[2] != "" {
  60. temp, err = strconv.ParseInt(strings.TrimPrefix(m[2], "."), 10, 32)
  61. if err != nil {
  62. return nil, fmt.Errorf("Error parsing version segment: %s", err)
  63. }
  64. sv.minor = temp
  65. } else {
  66. sv.minor = 0
  67. }
  68. if m[3] != "" {
  69. temp, err = strconv.ParseInt(strings.TrimPrefix(m[3], "."), 10, 32)
  70. if err != nil {
  71. return nil, fmt.Errorf("Error parsing version segment: %s", err)
  72. }
  73. sv.patch = temp
  74. } else {
  75. sv.patch = 0
  76. }
  77. return sv, nil
  78. }
  79. // MustParse parses a given version and panics on error.
  80. func MustParse(v string) *Version {
  81. sv, err := NewVersion(v)
  82. if err != nil {
  83. panic(err)
  84. }
  85. return sv
  86. }
  87. // String converts a Version object to a string.
  88. // Note, if the original version contained a leading v this version will not.
  89. // See the Original() method to retrieve the original value. Semantic Versions
  90. // don't contain a leading v per the spec. Instead it's optional on
  91. // impelementation.
  92. func (v *Version) String() string {
  93. var buf bytes.Buffer
  94. fmt.Fprintf(&buf, "%d.%d.%d", v.major, v.minor, v.patch)
  95. if v.pre != "" {
  96. fmt.Fprintf(&buf, "-%s", v.pre)
  97. }
  98. if v.metadata != "" {
  99. fmt.Fprintf(&buf, "+%s", v.metadata)
  100. }
  101. return buf.String()
  102. }
  103. // Original returns the original value passed in to be parsed.
  104. func (v *Version) Original() string {
  105. return v.original
  106. }
  107. // Major returns the major version.
  108. func (v *Version) Major() int64 {
  109. return v.major
  110. }
  111. // Minor returns the minor version.
  112. func (v *Version) Minor() int64 {
  113. return v.minor
  114. }
  115. // Patch returns the patch version.
  116. func (v *Version) Patch() int64 {
  117. return v.patch
  118. }
  119. // Prerelease returns the pre-release version.
  120. func (v *Version) Prerelease() string {
  121. return v.pre
  122. }
  123. // Metadata returns the metadata on the version.
  124. func (v *Version) Metadata() string {
  125. return v.metadata
  126. }
  127. // originalVPrefix returns the original 'v' prefix if any.
  128. func (v *Version) originalVPrefix() string {
  129. // Note, only lowercase v is supported as a prefix by the parser.
  130. if v.original != "" && v.original[:1] == "v" {
  131. return v.original[:1]
  132. }
  133. return ""
  134. }
  135. // IncPatch produces the next patch version.
  136. // If the current version does not have prerelease/metadata information,
  137. // it unsets metadata and prerelease values, increments patch number.
  138. // If the current version has any of prerelease or metadata information,
  139. // it unsets both values and keeps curent patch value
  140. func (v Version) IncPatch() Version {
  141. vNext := v
  142. // according to http://semver.org/#spec-item-9
  143. // Pre-release versions have a lower precedence than the associated normal version.
  144. // according to http://semver.org/#spec-item-10
  145. // Build metadata SHOULD be ignored when determining version precedence.
  146. if v.pre != "" {
  147. vNext.metadata = ""
  148. vNext.pre = ""
  149. } else {
  150. vNext.metadata = ""
  151. vNext.pre = ""
  152. vNext.patch = v.patch + 1
  153. }
  154. vNext.original = v.originalVPrefix() + "" + vNext.String()
  155. return vNext
  156. }
  157. // IncMinor produces the next minor version.
  158. // Sets patch to 0.
  159. // Increments minor number.
  160. // Unsets metadata.
  161. // Unsets prerelease status.
  162. func (v Version) IncMinor() Version {
  163. vNext := v
  164. vNext.metadata = ""
  165. vNext.pre = ""
  166. vNext.patch = 0
  167. vNext.minor = v.minor + 1
  168. vNext.original = v.originalVPrefix() + "" + vNext.String()
  169. return vNext
  170. }
  171. // IncMajor produces the next major version.
  172. // Sets patch to 0.
  173. // Sets minor to 0.
  174. // Increments major number.
  175. // Unsets metadata.
  176. // Unsets prerelease status.
  177. func (v Version) IncMajor() Version {
  178. vNext := v
  179. vNext.metadata = ""
  180. vNext.pre = ""
  181. vNext.patch = 0
  182. vNext.minor = 0
  183. vNext.major = v.major + 1
  184. vNext.original = v.originalVPrefix() + "" + vNext.String()
  185. return vNext
  186. }
  187. // SetPrerelease defines the prerelease value.
  188. // Value must not include the required 'hypen' prefix.
  189. func (v Version) SetPrerelease(prerelease string) (Version, error) {
  190. vNext := v
  191. if len(prerelease) > 0 && !validPrereleaseRegex.MatchString(prerelease) {
  192. return vNext, ErrInvalidPrerelease
  193. }
  194. vNext.pre = prerelease
  195. vNext.original = v.originalVPrefix() + "" + vNext.String()
  196. return vNext, nil
  197. }
  198. // SetMetadata defines metadata value.
  199. // Value must not include the required 'plus' prefix.
  200. func (v Version) SetMetadata(metadata string) (Version, error) {
  201. vNext := v
  202. if len(metadata) > 0 && !validPrereleaseRegex.MatchString(metadata) {
  203. return vNext, ErrInvalidMetadata
  204. }
  205. vNext.metadata = metadata
  206. vNext.original = v.originalVPrefix() + "" + vNext.String()
  207. return vNext, nil
  208. }
  209. // LessThan tests if one version is less than another one.
  210. func (v *Version) LessThan(o *Version) bool {
  211. return v.Compare(o) < 0
  212. }
  213. // GreaterThan tests if one version is greater than another one.
  214. func (v *Version) GreaterThan(o *Version) bool {
  215. return v.Compare(o) > 0
  216. }
  217. // Equal tests if two versions are equal to each other.
  218. // Note, versions can be equal with different metadata since metadata
  219. // is not considered part of the comparable version.
  220. func (v *Version) Equal(o *Version) bool {
  221. return v.Compare(o) == 0
  222. }
  223. // Compare compares this version to another one. It returns -1, 0, or 1 if
  224. // the version smaller, equal, or larger than the other version.
  225. //
  226. // Versions are compared by X.Y.Z. Build metadata is ignored. Prerelease is
  227. // lower than the version without a prerelease.
  228. func (v *Version) Compare(o *Version) int {
  229. // Compare the major, minor, and patch version for differences. If a
  230. // difference is found return the comparison.
  231. if d := compareSegment(v.Major(), o.Major()); d != 0 {
  232. return d
  233. }
  234. if d := compareSegment(v.Minor(), o.Minor()); d != 0 {
  235. return d
  236. }
  237. if d := compareSegment(v.Patch(), o.Patch()); d != 0 {
  238. return d
  239. }
  240. // At this point the major, minor, and patch versions are the same.
  241. ps := v.pre
  242. po := o.Prerelease()
  243. if ps == "" && po == "" {
  244. return 0
  245. }
  246. if ps == "" {
  247. return 1
  248. }
  249. if po == "" {
  250. return -1
  251. }
  252. return comparePrerelease(ps, po)
  253. }
  254. func compareSegment(v, o int64) int {
  255. if v < o {
  256. return -1
  257. }
  258. if v > o {
  259. return 1
  260. }
  261. return 0
  262. }
  263. func comparePrerelease(v, o string) int {
  264. // split the prelease versions by their part. The separator, per the spec,
  265. // is a .
  266. sparts := strings.Split(v, ".")
  267. oparts := strings.Split(o, ".")
  268. // Find the longer length of the parts to know how many loop iterations to
  269. // go through.
  270. slen := len(sparts)
  271. olen := len(oparts)
  272. l := slen
  273. if olen > slen {
  274. l = olen
  275. }
  276. // Iterate over each part of the prereleases to compare the differences.
  277. for i := 0; i < l; i++ {
  278. // Since the lentgh of the parts can be different we need to create
  279. // a placeholder. This is to avoid out of bounds issues.
  280. stemp := ""
  281. if i < slen {
  282. stemp = sparts[i]
  283. }
  284. otemp := ""
  285. if i < olen {
  286. otemp = oparts[i]
  287. }
  288. d := comparePrePart(stemp, otemp)
  289. if d != 0 {
  290. return d
  291. }
  292. }
  293. // Reaching here means two versions are of equal value but have different
  294. // metadata (the part following a +). They are not identical in string form
  295. // but the version comparison finds them to be equal.
  296. return 0
  297. }
  298. func comparePrePart(s, o string) int {
  299. // Fastpath if they are equal
  300. if s == o {
  301. return 0
  302. }
  303. // When s or o are empty we can use the other in an attempt to determine
  304. // the response.
  305. if o == "" {
  306. _, n := strconv.ParseInt(s, 10, 64)
  307. if n != nil {
  308. return -1
  309. }
  310. return 1
  311. }
  312. if s == "" {
  313. _, n := strconv.ParseInt(o, 10, 64)
  314. if n != nil {
  315. return 1
  316. }
  317. return -1
  318. }
  319. if s > o {
  320. return 1
  321. }
  322. return -1
  323. }