Class: SemanticPuppet::Version
- Inherits:
-
Numeric
- Object
- Numeric
- SemanticPuppet::Version
- Includes:
- Comparable
- Defined in:
- lib/semantic_puppet/version.rb
Overview
SemanticPuppet::Version subclasses Numeric so that it has sane Range semantics in Ruby 1.9+.
Defined Under Namespace
Classes: ValidationFailure
Constant Summary collapse
- REGEX_NUMERIC =
Version string matching regexes
'(0|[1-9]\d*)\.(0|[1-9]\d*)\.(0|[1-9]\d*)'.freeze
- REGEX_PRE =
Major . Minor . Patch
'(?:-([0-9A-Za-z-]+(?:\.[0-9A-Za-z-]+)*))?'.freeze
- REGEX_BUILD =
Prerelease
'(?:\+([0-9A-Za-z-]+(?:\.[0-9A-Za-z-]+)*))?'.freeze
- REGEX_FULL =
Build
REGEX_NUMERIC + REGEX_PRE + REGEX_BUILD.freeze
- REGEX_FULL_RX =
/\A#{REGEX_FULL}\Z/.freeze
- MIN =
The lowest precedence Version possible
self.new(0, 0, 0, [].freeze).freeze
- MAX =
The highest precedence Version possible
self.new(Float::INFINITY, 0, 0).freeze
Instance Attribute Summary collapse
-
#major ⇒ Object
readonly
Returns the value of attribute major.
-
#minor ⇒ Object
readonly
Returns the value of attribute minor.
-
#patch ⇒ Object
readonly
Returns the value of attribute patch.
Class Method Summary collapse
-
.parse(ver) ⇒ Version
Parse a Semantic Version string.
- .parse_build(build) ⇒ Object
- .parse_prerelease(prerelease) ⇒ Object
-
.valid?(ver) ⇒ bool
Validate a Semantic Version string.
Instance Method Summary collapse
- #<=>(other) ⇒ Object
-
#build ⇒ String
The
build
identifier as a string without the leading '+'. - #compare_prerelease(other) ⇒ Object
- #eql?(other) ⇒ Boolean (also: #==)
- #hash ⇒ Object
-
#initialize(major, minor, patch, prerelease = nil, build = nil) ⇒ Version
constructor
A new instance of Version.
- #next(part) ⇒ Object
-
#prerelease ⇒ String
The
prerelease
identifier as a string without the leading '-'. -
#stable? ⇒ Boolean
True if this is a stable release.
- #to_s ⇒ Object (also: #inspect)
-
#to_stable ⇒ Version
This version stripped from any prerelease identifier.
Constructor Details
#initialize(major, minor, patch, prerelease = nil, build = nil) ⇒ Version
Returns a new instance of Version.
55 56 57 58 59 60 61 |
# File 'lib/semantic_puppet/version.rb', line 55 def initialize(major, minor, patch, prerelease = nil, build = nil) @major = major @minor = minor @patch = patch @prerelease = prerelease @build = build end |
Instance Attribute Details
#major ⇒ Object (readonly)
Returns the value of attribute major.
53 54 55 |
# File 'lib/semantic_puppet/version.rb', line 53 def major @major end |
#minor ⇒ Object (readonly)
Returns the value of attribute minor.
53 54 55 |
# File 'lib/semantic_puppet/version.rb', line 53 def minor @minor end |
#patch ⇒ Object (readonly)
Returns the value of attribute patch.
53 54 55 |
# File 'lib/semantic_puppet/version.rb', line 53 def patch @patch end |
Class Method Details
.parse(ver) ⇒ Version
Parse a Semantic Version string.
15 16 17 18 19 20 21 |
# File 'lib/semantic_puppet/version.rb', line 15 def self.parse(ver) match, major, minor, patch, prerelease, build = *ver.match(REGEX_FULL_RX) raise ValidationFailure, "Unable to parse '#{ver}' as a semantic version identifier" unless match new(major.to_i, minor.to_i, patch.to_i, parse_prerelease(prerelease), parse_build(build)).freeze end |
.parse_build(build) ⇒ Object
37 38 39 |
# File 'lib/semantic_puppet/version.rb', line 37 def self.parse_build(build) build.nil? ? nil : build.split('.').freeze end |
.parse_prerelease(prerelease) ⇒ Object
41 42 43 44 45 46 47 48 49 50 51 |
# File 'lib/semantic_puppet/version.rb', line 41 def self.parse_prerelease(prerelease) return nil unless prerelease prerelease.split('.').map do |x| if x =~ /^\d+$/ raise ValidationFailure, 'Numeric pre-release identifiers MUST NOT contain leading zeroes' if x.length > 1 && x.start_with?('0') x.to_i else x end end.freeze end |
.valid?(ver) ⇒ bool
Validate a Semantic Version string.
27 28 29 30 31 32 33 34 35 |
# File 'lib/semantic_puppet/version.rb', line 27 def self.valid?(ver) match = ver.match(REGEX_FULL_RX) if match.nil? false else prerelease = match[4] prerelease.nil? || prerelease.split('.').all? { |x| !(x =~ /^0\d+$/) } end end |
Instance Method Details
#<=>(other) ⇒ Object
94 95 96 97 98 99 100 101 102 103 104 105 106 107 |
# File 'lib/semantic_puppet/version.rb', line 94 def <=>(other) return nil unless other.is_a?(Version) cmp = @major <=> other.major if cmp == 0 cmp = @minor <=> other.minor if cmp == 0 cmp = @patch <=> other.patch if cmp == 0 cmp = compare_prerelease(other) end end end cmp end |
#build ⇒ String
Returns the build
identifier as a string without the leading '+'.
90 91 92 |
# File 'lib/semantic_puppet/version.rb', line 90 def build (@build.nil? || @build.empty?) ? nil : @build.join('.') end |
#compare_prerelease(other) ⇒ Object
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 |
# File 'lib/semantic_puppet/version.rb', line 150 def compare_prerelease(other) mine = @prerelease # Need to use the instance variable here to avoid getting a string yours = other.instance_variable_get(:@prerelease) # A version that has a prerelease is always less than a version that doesn't if mine.nil? yours.nil? ? 0 : 1 elsif yours.nil? -1 else # Compare all prerelease identifier segments that can be compared. Should # all segments compare equal up to the point where one of the prereleases # have no more segments, then the one with more segements is greater. your_max = yours.size mine.each_with_index do |x, idx| # 'mine' win if 'your' list of segments is exhausted return 1 if idx >= your_max y = yours[idx] # Integer always wins over String cmp = if x.is_a?(Integer) y.is_a?(Integer) ? x <=> y : -1 elsif y.is_a?(Integer) 1 else x <=> y end # No need to continue if a diff is found return cmp unless cmp == 0 end # All segments, up to the point where at least one list of segement is exhausted, # compared equal. The one with the highest segment count wins. mine.size <=> your_max end end |
#eql?(other) ⇒ Boolean Also known as: ==
109 110 111 112 113 114 115 116 |
# File 'lib/semantic_puppet/version.rb', line 109 def eql?(other) other.is_a?(Version) && @major.eql?(other.major) && @minor.eql?(other.minor) && @patch.eql?(other.patch) && @prerelease.eql?(other.instance_variable_get(:@prerelease)) && @build.eql?(other.instance_variable_get(:@build)) end |
#hash ⇒ Object
146 147 148 |
# File 'lib/semantic_puppet/version.rb', line 146 def hash (((((@major * 0x100) ^ @minor) * 0x100) ^ @patch) * 0x100) ^ @prerelease.hash end |
#next(part) ⇒ Object
63 64 65 66 67 68 69 70 71 72 |
# File 'lib/semantic_puppet/version.rb', line 63 def next(part) case part when :major self.class.new(@major.next, 0, 0) when :minor self.class.new(@major, @minor.next, 0) when :patch self.class.new(@major, @minor, @patch.next) end end |
#prerelease ⇒ String
Returns the prerelease
identifier as a string without the leading '-'.
75 76 77 |
# File 'lib/semantic_puppet/version.rb', line 75 def prerelease (@prerelease.nil? || @prerelease.empty?) ? nil : @prerelease.join('.') end |
#stable? ⇒ Boolean
Returns true if this is a stable release.
80 81 82 |
# File 'lib/semantic_puppet/version.rb', line 80 def stable? @prerelease.nil? || @prerelease.empty? end |
#to_s ⇒ Object Also known as: inspect
119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 |
# File 'lib/semantic_puppet/version.rb', line 119 def to_s s = "#{@major}.#{@minor}.#{@patch}" # Appending the @prerelease and @build in a thoroughly tested and optimized # way. Using interpolations and/or array joins may look simpler but will slows # things down. Don't change this code without measuring performance of new # solution. unless @prerelease.nil? s << '-' << @prerelease[0].to_s i = 0 l = @prerelease.length while (i += 1) < l s << '.' << @prerelease[i].to_s end end unless @build.nil? s << '+' << @build[0].to_s i = 0 l = @build.length while (i += 1) < l s << '.' << @build[i].to_s end end s end |