|  | // Copyright 2020 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.util; | 
|  |  | 
|  | import static com.google.common.base.Preconditions.checkArgument; | 
|  | import static com.google.common.base.Preconditions.checkNotNull; | 
|  |  | 
|  | import com.google.devtools.build.lib.server.FailureDetails; | 
|  | import com.google.devtools.build.lib.server.FailureDetails.FailureDetail; | 
|  | import com.google.protobuf.Descriptors.EnumValueDescriptor; | 
|  | import com.google.protobuf.Descriptors.FieldDescriptor; | 
|  | import com.google.protobuf.MessageOrBuilder; | 
|  | import java.util.Comparator; | 
|  | import java.util.Map; | 
|  | import java.util.Objects; | 
|  | import javax.annotation.Nullable; | 
|  |  | 
|  | /** An {@link ExitCode} that has a {@link FailureDetail} unless it's {@link ExitCode#SUCCESS}. */ | 
|  | public class DetailedExitCode { | 
|  | private final ExitCode exitCode; | 
|  | @Nullable private final FailureDetail failureDetail; | 
|  |  | 
|  | private DetailedExitCode(ExitCode exitCode, @Nullable FailureDetail failureDetail) { | 
|  | this.exitCode = exitCode; | 
|  | this.failureDetail = failureDetail; | 
|  | } | 
|  |  | 
|  | public ExitCode getExitCode() { | 
|  | return exitCode; | 
|  | } | 
|  |  | 
|  | /** Returns the registered {@link ExitCode} associated with a {@link FailureDetail} message. */ | 
|  | public static ExitCode getExitCode(FailureDetail failureDetail) { | 
|  | // TODO(mschaller): Consider specializing for unregistered exit codes here, if absolutely | 
|  | //  necessary. | 
|  | int numericExitCode = getNumericExitCode(failureDetail); | 
|  | return checkNotNull( | 
|  | ExitCode.forCode(numericExitCode), "No ExitCode for numericExitCode %s", numericExitCode); | 
|  | } | 
|  |  | 
|  | @Nullable | 
|  | public FailureDetail getFailureDetail() { | 
|  | return failureDetail; | 
|  | } | 
|  |  | 
|  | public boolean isSuccess() { | 
|  | return exitCode.equals(ExitCode.SUCCESS); | 
|  | } | 
|  |  | 
|  | /** Returns a {@link DetailedExitCode} specifying success (i.e. exit code 0). */ | 
|  | public static DetailedExitCode success() { | 
|  | return new DetailedExitCode(ExitCode.SUCCESS, null); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a {@link DetailedExitCode} combining the provided {@link FailureDetail} and {@link | 
|  | * ExitCode}. | 
|  | * | 
|  | * <p>This method exists in order to allow for the introduction of new {@link | 
|  | * FailureDetail)-handling code infrastructure without requiring any simultaneous change in exit | 
|  | * code behavior. | 
|  | * | 
|  | * <p>Unless contrained by backwards-compatibility needs, callers should use {@link | 
|  | * #of(FailureDetail)} instead. | 
|  | */ | 
|  | public static DetailedExitCode of(ExitCode exitCode, FailureDetail failureDetail) { | 
|  | return new DetailedExitCode(checkNotNull(exitCode), checkNotNull(failureDetail)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a {@link DetailedExitCode} whose {@link ExitCode} is chosen referencing {@link | 
|  | * FailureDetail}'s metadata. | 
|  | */ | 
|  | public static DetailedExitCode of(FailureDetail failureDetail) { | 
|  | return new DetailedExitCode(getExitCode(failureDetail), checkNotNull(failureDetail)); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int hashCode() { | 
|  | return Objects.hash(exitCode, failureDetail); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public boolean equals(Object obj) { | 
|  | if (obj == this) { | 
|  | return true; | 
|  | } | 
|  | if (!(obj instanceof DetailedExitCode)) { | 
|  | return false; | 
|  | } | 
|  | DetailedExitCode that = (DetailedExitCode) obj; | 
|  | return this.exitCode.equals(that.exitCode) | 
|  | && Objects.equals(this.failureDetail, that.failureDetail); | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String toString() { | 
|  | return String.format( | 
|  | "DetailedExitCode{exitCode=%s, failureDetail=%s}", exitCode, failureDetail); | 
|  | } | 
|  |  | 
|  | /** Returns the numeric exit code associated with a {@link FailureDetail} message. */ | 
|  | private static int getNumericExitCode(FailureDetail failureDetail) { | 
|  | MessageOrBuilder categoryMsg = getCategorySubmessage(failureDetail); | 
|  | EnumValueDescriptor subcategoryDescriptor = | 
|  | getSubcategoryDescriptor(failureDetail, categoryMsg); | 
|  | return getNumericExitCode(subcategoryDescriptor); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the numeric exit code associated with a {@link FailureDetail} submessage's subcategory | 
|  | * enum value. | 
|  | */ | 
|  | public static int getNumericExitCode(EnumValueDescriptor subcategoryDescriptor) { | 
|  | checkArgument( | 
|  | subcategoryDescriptor.getOptions().hasExtension(FailureDetails.metadata), | 
|  | "Enum value %s has no FailureDetails.metadata", | 
|  | subcategoryDescriptor); | 
|  | return subcategoryDescriptor.getOptions().getExtension(FailureDetails.metadata).getExitCode(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the category submessage, i.e. the message in {@link FailureDetail}'s oneof. Throws if | 
|  | * none of those fields are set. | 
|  | */ | 
|  | private static MessageOrBuilder getCategorySubmessage(FailureDetail failureDetail) { | 
|  | MessageOrBuilder categoryMsg = null; | 
|  | for (Map.Entry<FieldDescriptor, Object> entry : failureDetail.getAllFields().entrySet()) { | 
|  | FieldDescriptor fieldDescriptor = entry.getKey(); | 
|  | if (isCategoryField(fieldDescriptor)) { | 
|  | categoryMsg = (MessageOrBuilder) entry.getValue(); | 
|  | break; | 
|  | } | 
|  | } | 
|  | return checkNotNull( | 
|  | categoryMsg, "FailureDetail missing category submessage: %s", failureDetail); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns whether the {@link FieldDescriptor} describes a field in {@link FailureDetail}'s oneof. | 
|  | * | 
|  | * <p>Uses the field number criteria described in failure_details.proto. | 
|  | */ | 
|  | private static boolean isCategoryField(FieldDescriptor fieldDescriptor) { | 
|  | int fieldNum = fieldDescriptor.getNumber(); | 
|  | return 100 < fieldNum && fieldNum <= 10_000; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the enum value descriptor for the enum field with field number 1 in the {@link | 
|  | * FailureDetail}'s category submessage. | 
|  | */ | 
|  | private static EnumValueDescriptor getSubcategoryDescriptor( | 
|  | FailureDetail failureDetail, MessageOrBuilder categoryMsg) { | 
|  | FieldDescriptor fieldNumberOne = categoryMsg.getDescriptorForType().findFieldByNumber(1); | 
|  | checkNotNull( | 
|  | fieldNumberOne, "FailureDetail category submessage has no field #1: %s", failureDetail); | 
|  | Object fieldNumberOneVal = categoryMsg.getField(fieldNumberOne); | 
|  | checkArgument( | 
|  | fieldNumberOneVal instanceof EnumValueDescriptor, | 
|  | "FailureDetail category submessage has non-enum field #1: %s", | 
|  | failureDetail); | 
|  | return (EnumValueDescriptor) fieldNumberOneVal; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * A comparator to determine the reporting priority of {@link DetailedExitCode}. | 
|  | * | 
|  | * <p>Priority: infrastructure exit codes > non-infrastructure exit codes > null exit codes, with | 
|  | * exit codes that contain failure details taking priority within each class. | 
|  | */ | 
|  | public static class DetailedExitCodeComparator implements Comparator<DetailedExitCode> { | 
|  | public static final DetailedExitCodeComparator INSTANCE = new DetailedExitCodeComparator(); | 
|  |  | 
|  | private DetailedExitCodeComparator() {} | 
|  |  | 
|  | @Nullable | 
|  | public static DetailedExitCode chooseMoreImportantWithFirstIfTie( | 
|  | @Nullable DetailedExitCode first, @Nullable DetailedExitCode second) { | 
|  | return INSTANCE.compare(first, second) >= 0 ? first : second; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public int compare(DetailedExitCode c1, DetailedExitCode c2) { | 
|  | // returns POSITIVE result when the priority of c1 is HIGHER than the priority of c2 | 
|  | return getPriority(c1) - getPriority(c2); | 
|  | } | 
|  |  | 
|  | private static int getPriority(DetailedExitCode code) { | 
|  | if (code == null) { | 
|  | return 0; | 
|  | } else { | 
|  | int codeClass = code.getExitCode().isInfrastructureFailure() ? 4 : 2; | 
|  | return codeClass + (code.getFailureDetail() != null ? 1 : 0); | 
|  | } | 
|  | } | 
|  | } | 
|  | } |