| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269 | package versionimport (	"regexp"	"strconv"	"strings")type ConstraintGroup struct {	constraints []*Constraint}// Return a new NewConstrainGroupfunc NewConstrainGroup() *ConstraintGroup {	group := new(ConstraintGroup)	return group}// Return a new NewConstrainGroup and create the constraints based on a string//// Version constraints can be specified in a few different ways://// Exact version: You can specify the exact version of a package, for // example 1.0.2.//// Range: By using comparison operators you can specify ranges of valid versions. // Valid operators are >, >=, <, <=, !=. An example range would be >=1.0. You can // define multiple ranges, separated by a comma: >=1.0,<2.0.// // Wildcard: You can specify a pattern with a * wildcard. 1.0.* is the equivalent // of >=1.0,<1.1.//// Next Significant Release (Tilde Operator): The ~ operator is best explained by // example: ~1.2 is equivalent to >=1.2,<2.0, while ~1.2.3 is equivalent to // >=1.2.3,<1.3. As you can see it is mostly useful for projects respecting // semantic versioning. A common usage would be to mark the minimum minor // version you depend on, like ~1.2 (which allows anything up to, but not // including, 2.0). Since in theory there should be no backwards compatibility // breaks until 2.0, that works well. Another way of looking at it is that // using ~ specifies a minimum version, but allows the last digit specified // to go up.//// By default only stable releases are taken into consideration. If you would like // to also get RC, beta, alpha or dev versions of your dependencies you can do so // using stability flags. To change that for all packages instead of doing per // dependency you can also use the minimum-stability setting.// // From: http://getcomposer.org/doc/01-basic-usage.md#package-versionsfunc NewConstrainGroupFromString(name string) *ConstraintGroup {	group := new(ConstraintGroup)	group.fromString(name)	return group}// Adds a Contraint to the groupfunc (self *ConstraintGroup) AddConstraint(constraint ...*Constraint) {	if self.constraints == nil {		self.constraints = make([]*Constraint, 0)	}	self.constraints = append(self.constraints, constraint...)}// Return all the constraintsfunc (self *ConstraintGroup) GetConstraints() []*Constraint {	return self.constraints}// Match a given version againts the group//// Usage//     c := version.NewConstrainGroupFromString(">2.0,<=3.0")//     c.Match("2.5.0beta")//     Returns: true////     c := version.NewConstrainGroupFromString("~1.2.3")//     c.Match("1.2.3.5")//     Returns: truefunc (self *ConstraintGroup) Match(version string) bool {	for _, constraint := range self.constraints {		if constraint.Match(version) == false {			return false		}	}	return true}func (self *ConstraintGroup) fromString(constraint string) bool {	result := RegFind(`(?i)^([^,\s]*?)@(stable|RC|beta|alpha|dev)$`, constraint)	if result != nil {		constraint = result[1]		if constraint == "" {			constraint = "*"		}	}	result = RegFind(`(?i)^(dev-[^,\s@]+?|[^,\s@]+?\.x-dev)#.+$`, constraint)	if result != nil {		if result[1] != "" {			constraint = result[1]		}	}	constraints := RegSplit(`\s*,\s*`, strings.Trim(constraint, " "))	if len(constraints) > 1 {		for _, part := range constraints {			self.AddConstraint(self.parseConstraint(part)...)		}		return true	}	self.AddConstraint(self.parseConstraint(constraints[0])...)	return true}func (self *ConstraintGroup) parseConstraint(constraint string) []*Constraint {	stabilityModifier := ""	result := RegFind(`(?i)^([^,\s]+?)@(stable|RC|beta|alpha|dev)$`, constraint)	if result != nil {		constraint = result[1]		if result[2] != "stable" {			stabilityModifier = result[2]		}	}	result = RegFind(`^[x*](\.[x*])*$`, constraint)	if result != nil {		return make([]*Constraint, 0)	}	highVersion := ""	lowVersion := ""	result = RegFind(`(?i)^~(\d+)(?:\.(\d+))?(?:\.(\d+))?(?:\.(\d+))?`+modifierRegex+`?$`, constraint)	if result != nil {		if len(result) > 4 && result[4] != "" {			last, _ := strconv.Atoi(result[3])			highVersion = result[1] + "." + result[2] + "." + strconv.Itoa(last+1) + ".0-dev"			lowVersion = result[1] + "." + result[2] + "." + result[3] + "." + result[4]		} else if len(result) > 3 && result[3] != "" {			last, _ := strconv.Atoi(result[2])			highVersion = result[1] + "." + strconv.Itoa(last+1) + ".0.0-dev"			lowVersion = result[1] + "." + result[2] + "." + result[3] + ".0"		} else {			last, _ := strconv.Atoi(result[1])			highVersion = strconv.Itoa(last+1) + ".0.0.0-dev"			if len(result) > 2 && result[2] != "" {				lowVersion = result[1] + "." + result[2] + ".0.0"			} else {				lowVersion = result[1] + ".0.0.0"			}		}		if len(result) > 5 && result[5] != "" {			lowVersion = lowVersion + "-" + expandStability(result[5])		}		if len(result) > 6 && result[6] != "" {			lowVersion = lowVersion + result[6]		}		if len(result) > 7 && result[7] != "" {			lowVersion = lowVersion + "-dev"		}		return []*Constraint{			{">=", lowVersion},			{"<", highVersion},		}	}	result = RegFind(`^(\d+)(?:\.(\d+))?(?:\.(\d+))?\.[x*]$`, constraint)	if result != nil {		if len(result) > 3 && result[3] != "" {			highVersion = result[1] + "." + result[2] + "." + result[3] + ".9999999"			if result[3] == "0" {				last, _ := strconv.Atoi(result[2])				lowVersion = result[1] + "." + strconv.Itoa(last-1) + ".9999999.9999999"			} else {				last, _ := strconv.Atoi(result[3])				lowVersion = result[1] + "." + result[2] + "." + strconv.Itoa(last-1) + ".9999999"			}		} else if len(result) > 2 && result[2] != "" {			highVersion = result[1] + "." + result[2] + ".9999999.9999999"			if result[2] == "0" {				last, _ := strconv.Atoi(result[1])				lowVersion = strconv.Itoa(last-1) + ".9999999.9999999.9999999"			} else {				last, _ := strconv.Atoi(result[2])				lowVersion = result[1] + "." + strconv.Itoa(last-1) + ".9999999.9999999"			}		} else {			highVersion = result[1] + ".9999999.9999999.9999999"			if result[1] == "0" {				return []*Constraint{{"<", highVersion}}			} else {				last, _ := strconv.Atoi(result[1])				lowVersion = strconv.Itoa(last-1) + ".9999999.9999999.9999999"			}		}		return []*Constraint{			{">", lowVersion},			{"<", highVersion},		}	}	// match operators constraints	result = RegFind(`^(<>|!=|>=?|<=?|==?)?\s*(.*)`, constraint)	if result != nil {		version := Normalize(result[2])		if stabilityModifier != "" && parseStability(version) == "stable" {			version = version + "-" + stabilityModifier		} else if result[1] == "<" {			match := RegFind(`(?i)-stable$`, result[2])			if match == nil {				version = version + "-dev"			}		}		if len(result) > 1 && result[1] != "" {			return []*Constraint{{result[1], version}}		} else {			return []*Constraint{{"=", version}}		}	}	return []*Constraint{{constraint, stabilityModifier}}}func RegFind(pattern, subject string) []string {	reg := regexp.MustCompile(pattern)	matched := reg.FindAllStringSubmatch(subject, -1)	if matched != nil {		return matched[0]	}	return nil}func RegSplit(pattern, subject string) []string {	reg := regexp.MustCompile(pattern)	indexes := reg.FindAllStringIndex(subject, -1)	laststart := 0	result := make([]string, len(indexes)+1)	for i, element := range indexes {		result[i] = subject[laststart:element[0]]		laststart = element[1]	}	result[len(indexes)] = subject[laststart:len(subject)]	return result}
 |