blob: 7661a7d0b89eb95870c70b81e3a74d965d28956e [file] [log] [blame]
// Copyright 2019 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.versioning;
import static com.google.common.base.Preconditions.checkArgument;
import com.google.auto.value.AutoValue;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* Represents semantic versions (semver) and allows parsing them.
*
* <p>This implementation is supposed to be program-agnostic and thus everything it supports should
* be applicable to any program following semver.
*/
// TODO(jmmv): Would be nice to take com.google.devtools.build.lib.rules.apple.DottedVersion and
// build the oddities of that scheme on top of this, given that the differences are minimal.
@AutoValue
public abstract class SemVer implements Comparable<SemVer> {
/** The major version stored in this semver instance. */
abstract int major();
/** The minor version stored in this semver instance. */
abstract int minor();
/** The patch version stored in this semver instance. */
abstract int patch();
/** Constructs a new semver from the given components with minor and patch set to 0. */
public static SemVer from(int major) {
return from(major, 0, 0);
}
/** Constructs a new semver from the given components with patch set to 0. */
public static SemVer from(int major, int minor) {
return from(major, minor, 0);
}
/** Constructs a new semver from the given components. */
public static SemVer from(int major, int minor, int patch) {
checkArgument(major >= 0);
checkArgument(minor >= 0);
checkArgument(patch >= 0);
return new AutoValue_SemVer(major, minor, patch);
}
/** Parses a semver from a string. */
public static SemVer parse(String text) throws ParseException {
Pattern pattern = Pattern.compile("([0-9]+)(.([0-9]+)(.([0-9]+))?)?");
Matcher matcher = pattern.matcher(text);
if (!matcher.matches()) {
throw new ParseException("Invalid semver " + text);
}
try {
int major = Integer.parseInt(matcher.group(1));
String maybeMinor = matcher.group(3);
String maybePatch = matcher.group(5);
if (maybePatch != null) {
return from(major, Integer.parseInt(maybeMinor), Integer.parseInt(maybePatch));
} else if (matcher.group(3) != null) {
return from(major, Integer.parseInt(maybeMinor));
} else {
return from(major);
}
} catch (NumberFormatException e) {
throw new ParseException("Invalid number in semver component", e);
}
}
@Override
public int compareTo(SemVer o) {
int majorRelation = Integer.compare(major(), o.major());
int minorRelation = Integer.compare(minor(), o.minor());
int patchRelation = Integer.compare(patch(), o.patch());
return majorRelation != 0
? majorRelation
: (minorRelation != 0 ? minorRelation : patchRelation);
}
@Override
public final String toString() {
return String.format("%d.%d.%d", major(), minor(), patch());
}
}