|  | // Copyright 2018 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.remote; | 
|  |  | 
|  | import build.bazel.remote.execution.v2.ServerCapabilities; | 
|  | import build.bazel.semver.SemVer; | 
|  |  | 
|  | /** | 
|  | * Represents a version of the Remote Execution API. | 
|  | */ | 
|  | public class ApiVersion implements Comparable<ApiVersion> { | 
|  | public final int major; | 
|  | public final int minor; | 
|  | public final int patch; | 
|  | public final String prerelease; | 
|  |  | 
|  | // The current version of the Remote Execution API. This field will need to be updated | 
|  | // together with all version changes. | 
|  | public static final ApiVersion current = new ApiVersion(SemVer.newBuilder().setMajor(2).build()); | 
|  |  | 
|  | public ApiVersion(int major, int minor, int patch, String prerelease) { | 
|  | this.major = major; | 
|  | this.minor = minor; | 
|  | this.patch = patch; | 
|  | this.prerelease = prerelease; | 
|  | } | 
|  |  | 
|  | public ApiVersion(SemVer semver) { | 
|  | this(semver.getMajor(), semver.getMinor(), semver.getPatch(), semver.getPrerelease()); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String toString() { | 
|  | if (!prerelease.isEmpty()) { | 
|  | return prerelease; | 
|  | } | 
|  | StringBuilder builder = new StringBuilder(); | 
|  | builder.append(major); | 
|  | builder.append("."); | 
|  | builder.append(minor); | 
|  | if (patch != 0) { | 
|  | builder.append("."); | 
|  | builder.append(patch); | 
|  | } | 
|  | return builder.toString(); | 
|  | } | 
|  |  | 
|  | public SemVer toSemVer() { | 
|  | return SemVer.newBuilder() | 
|  | .setMajor(major) | 
|  | .setMinor(minor) | 
|  | .setPatch(patch) | 
|  | .setPrerelease(prerelease) | 
|  | .build(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Compares the current API version to another API version. | 
|  | * | 
|  | * @param other the API version to compare to. | 
|  | * @return 0 if the API versions are equal, a number less than 0 if the current version is earlier | 
|  | *     than other, and a number greater than 0 if the current version is later than other. It is | 
|  | *     assumed that all prerelease versions are earlier than all released versions. | 
|  | */ | 
|  | @Override | 
|  | public int compareTo(ApiVersion other) { | 
|  | if (!prerelease.isEmpty()) { | 
|  | if (other.prerelease.isEmpty()) { | 
|  | return -1; | 
|  | } | 
|  | return prerelease.compareTo(other.prerelease); | 
|  | } | 
|  | if (!other.prerelease.isEmpty()) { | 
|  | return 1; | 
|  | } | 
|  | if (major != other.major) { | 
|  | return Integer.compare(major, other.major); | 
|  | } | 
|  | if (minor != other.minor) { | 
|  | return Integer.compare(minor, other.minor); | 
|  | } | 
|  | return Integer.compare(patch, other.patch); | 
|  | } | 
|  |  | 
|  | static class ServerSupportedStatus { | 
|  | private enum State { | 
|  | SUPPORTED, | 
|  | UNSUPPORTED, | 
|  | DEPRECATED, | 
|  | } | 
|  | private final String message; | 
|  | private final State state; | 
|  |  | 
|  | private ServerSupportedStatus(State state, String message) { | 
|  | this.state = state; | 
|  | this.message = message; | 
|  | } | 
|  |  | 
|  | public static ServerSupportedStatus supported() { | 
|  | return new ServerSupportedStatus(State.SUPPORTED, ""); | 
|  | } | 
|  |  | 
|  | public static ServerSupportedStatus unsupported( | 
|  | ApiVersion curr, ApiVersion lowApiVersion, ApiVersion highApiVersion) { | 
|  | return new ServerSupportedStatus( | 
|  | State.UNSUPPORTED, | 
|  | String.format( | 
|  | "The API version %s is not supported by the server. " | 
|  | + "Please switch to a supported version: %s to %s.", | 
|  | curr, lowApiVersion, highApiVersion)); | 
|  | } | 
|  |  | 
|  | public static ServerSupportedStatus deprecated( | 
|  | ApiVersion curr, ApiVersion lowApiVersion, ApiVersion highApiVersion) { | 
|  | return new ServerSupportedStatus( | 
|  | State.DEPRECATED, | 
|  | String.format( | 
|  | "The API version %s is deprecated by the server. " | 
|  | + "Please upgrade to a recommended version: %s to %s.", | 
|  | curr, lowApiVersion, highApiVersion)); | 
|  | } | 
|  |  | 
|  | public String getMessage() { | 
|  | return message; | 
|  | } | 
|  |  | 
|  | public boolean isSupported() { | 
|  | return state == State.SUPPORTED; | 
|  | } | 
|  |  | 
|  | public boolean isDeprecated() { | 
|  | return state == State.DEPRECATED; | 
|  | } | 
|  |  | 
|  | public boolean isUnsupported() { | 
|  | return state == State.UNSUPPORTED; | 
|  | } | 
|  | } | 
|  |  | 
|  | public ServerSupportedStatus checkServerSupportedVersions(ServerCapabilities cap) { | 
|  | ApiVersion deprecated = | 
|  | cap.hasDeprecatedApiVersion() ? new ApiVersion(cap.getDeprecatedApiVersion()) : null; | 
|  | ApiVersion low = new ApiVersion(cap.getLowApiVersion()); | 
|  | ApiVersion high = new ApiVersion(cap.getHighApiVersion()); | 
|  | if (deprecated != null && compareTo(deprecated) >= 0 && compareTo(low) < 0) { | 
|  | return ServerSupportedStatus.deprecated(this, low, high); | 
|  | } | 
|  | if (compareTo(low) < 0 || compareTo(high) > 0) { | 
|  | return ServerSupportedStatus.unsupported(this, low, high); | 
|  | } | 
|  | return ServerSupportedStatus.supported(); | 
|  | } | 
|  | } |