// Copyright 2014 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.skyframe; import com.google.common.annotations.VisibleForTesting; import javax.annotation.Nullable; /** Utilities for producing and consuming ValueOrException(2|3|4)? instances. */ @VisibleForTesting public class ValueOrExceptionUtils { /** The bottom exception type. */ class BottomException extends Exception { } @Nullable public static SkyValue downconvert(ValueOrException voe) { return voe.getValue(); } public static ValueOrException downconvert( ValueOrException2 voe, Class exceptionClass1) { Exception e = voe.getException(); if (e == null) { return new ValueOrExceptionValueImpl<>(voe.getValue()); } // Here and below, we use type-safe casts for performance reasons. Another approach would be // cascading try-catch-rethrow blocks, but that has a higher performance penalty. if (exceptionClass1.isInstance(e)) { return new ValueOrExceptionExnImpl<>(exceptionClass1.cast(e)); } throw new IllegalStateException("shouldn't reach here " + e.getClass() + " " + exceptionClass1, e); } public static ValueOrException2 downconvert( ValueOrException3 voe, Class exceptionClass1, Class exceptionClass2) { Exception e = voe.getException(); if (e == null) { return new ValueOrException2ValueImpl<>(voe.getValue()); } if (exceptionClass1.isInstance(e)) { return new ValueOrException2Exn1Impl<>(exceptionClass1.cast(e)); } if (exceptionClass2.isInstance(e)) { return new ValueOrException2Exn2Impl<>(exceptionClass2.cast(e)); } throw new IllegalStateException("shouldn't reach here " + e.getClass() + " " + exceptionClass1 + " " + exceptionClass2, e); } public static ValueOrException3 downconvert( ValueOrException4 voe, Class exceptionClass1, Class exceptionClass2, Class exceptionClass3) { Exception e = voe.getException(); if (e == null) { return new ValueOrException3ValueImpl<>(voe.getValue()); } if (exceptionClass1.isInstance(e)) { return new ValueOrException3Exn1Impl<>(exceptionClass1.cast(e)); } if (exceptionClass2.isInstance(e)) { return new ValueOrException3Exn2Impl<>(exceptionClass2.cast(e)); } if (exceptionClass3.isInstance(e)) { return new ValueOrException3Exn3Impl<>(exceptionClass3.cast(e)); } throw new IllegalStateException("shouldn't reach here " + e.getClass() + " " + exceptionClass1 + " " + exceptionClass2 + " " + exceptionClass3, e); } public static ValueOrException4 downconvert( ValueOrException5 voe, Class exceptionClass1, Class exceptionClass2, Class exceptionClass3, Class exceptionClass4) { Exception e = voe.getException(); if (e == null) { return new ValueOrException4ValueImpl<>(voe.getValue()); } if (exceptionClass1.isInstance(e)) { return new ValueOrException4Exn1Impl<>(exceptionClass1.cast(e)); } if (exceptionClass2.isInstance(e)) { return new ValueOrException4Exn2Impl<>(exceptionClass2.cast(e)); } if (exceptionClass3.isInstance(e)) { return new ValueOrException4Exn3Impl<>(exceptionClass3.cast(e)); } if (exceptionClass4.isInstance(e)) { return new ValueOrException4Exn4Impl<>(exceptionClass4.cast(e)); } throw new IllegalStateException("shouldn't reach here " + e.getClass() + " " + exceptionClass1 + " " + exceptionClass2 + " " + exceptionClass3 + " " + exceptionClass4, e); } public static ValueOrException ofNull() { return ValueOrExceptionValueImpl.ofNull(); } public static ValueOrUntypedException ofValueUntyped(SkyValue value) { return new ValueOrUntypedExceptionImpl(value); } public static ValueOrException ofExn(E e) { return new ValueOrExceptionExnImpl<>(e); } public static ValueOrException5 ofNullValue() { return ValueOrException5ValueImpl.ofNullValue(); } public static ValueOrException5 ofValue(SkyValue value) { return new ValueOrException5ValueImpl<>(value); } public static ValueOrException5 ofExn1(E1 e) { return new ValueOrException5Exn1Impl<>(e); } public static ValueOrException5 ofExn2(E2 e) { return new ValueOrException5Exn2Impl<>(e); } public static ValueOrException5 ofExn3(E3 e) { return new ValueOrException5Exn3Impl<>(e); } public static ValueOrException5 ofExn4(E4 e) { return new ValueOrException5Exn4Impl<>(e); } public static ValueOrException5 ofExn5(E5 e) { return new ValueOrException5Exn5Impl<>(e); } private static class ValueOrUntypedExceptionImpl extends ValueOrUntypedException { @Nullable private final SkyValue value; ValueOrUntypedExceptionImpl(@Nullable SkyValue value) { this.value = value; } @Override @Nullable public SkyValue getValue() { return value; } @Override public Exception getException() { return null; } } private static class ValueOrExceptionValueImpl extends ValueOrException { private static final ValueOrExceptionValueImpl NULL = new ValueOrExceptionValueImpl((SkyValue) null); @Nullable private final SkyValue value; private ValueOrExceptionValueImpl(@Nullable SkyValue value) { this.value = value; } @Override @Nullable public SkyValue get() { return value; } @Override @Nullable public SkyValue getValue() { return value; } @Override @Nullable public Exception getException() { return null; } @SuppressWarnings("unchecked") public static ValueOrExceptionValueImpl ofNull() { return (ValueOrExceptionValueImpl) NULL; } } private static class ValueOrExceptionExnImpl extends ValueOrException { private final E e; private ValueOrExceptionExnImpl(E e) { this.e = e; } @Override public SkyValue get() throws E { throw e; } @Override @Nullable public SkyValue getValue() { return null; } @Override public Exception getException() { return e; } } private static class ValueOrException2ValueImpl extends ValueOrException2 { @Nullable private final SkyValue value; ValueOrException2ValueImpl(@Nullable SkyValue value) { this.value = value; } @Override @Nullable public SkyValue get() { return value; } @Override @Nullable public Exception getException() { return null; } @Override @Nullable public SkyValue getValue() { return value; } } private static class ValueOrException2Exn1Impl extends ValueOrException2 { private final E1 e; private ValueOrException2Exn1Impl(E1 e) { this.e = e; } @Override public SkyValue get() throws E1 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException2Exn2Impl extends ValueOrException2 { private final E2 e; private ValueOrException2Exn2Impl(E2 e) { this.e = e; } @Override public SkyValue get() throws E2 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException3ValueImpl extends ValueOrException3 { @Nullable private final SkyValue value; ValueOrException3ValueImpl(@Nullable SkyValue value) { this.value = value; } @Override @Nullable public SkyValue get() { return value; } @Override @Nullable public Exception getException() { return null; } @Override @Nullable public SkyValue getValue() { return value; } } private static class ValueOrException3Exn1Impl extends ValueOrException3 { private final E1 e; private ValueOrException3Exn1Impl(E1 e) { this.e = e; } @Override public SkyValue get() throws E1 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException3Exn2Impl extends ValueOrException3 { private final E2 e; private ValueOrException3Exn2Impl(E2 e) { this.e = e; } @Override public SkyValue get() throws E2 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException3Exn3Impl extends ValueOrException3 { private final E3 e; private ValueOrException3Exn3Impl(E3 e) { this.e = e; } @Override public SkyValue get() throws E3 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException4ValueImpl extends ValueOrException4 { @Nullable private final SkyValue value; ValueOrException4ValueImpl(@Nullable SkyValue value) { this.value = value; } @Override @Nullable public SkyValue get() { return value; } @Override @Nullable public Exception getException() { return null; } @Override @Nullable public SkyValue getValue() { return value; } } private static class ValueOrException4Exn1Impl extends ValueOrException4 { private final E1 e; private ValueOrException4Exn1Impl(E1 e) { this.e = e; } @Override public SkyValue get() throws E1 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException4Exn2Impl extends ValueOrException4 { private final E2 e; private ValueOrException4Exn2Impl(E2 e) { this.e = e; } @Override public SkyValue get() throws E2 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException4Exn3Impl extends ValueOrException4 { private final E3 e; private ValueOrException4Exn3Impl(E3 e) { this.e = e; } @Override public SkyValue get() throws E3 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException4Exn4Impl extends ValueOrException4 { private final E4 e; private ValueOrException4Exn4Impl(E4 e) { this.e = e; } @Override public SkyValue get() throws E4 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException5ValueImpl extends ValueOrException5 { private static final ValueOrException5ValueImpl NULL = new ValueOrException5ValueImpl<>((SkyValue) null); @Nullable private final SkyValue value; ValueOrException5ValueImpl(@Nullable SkyValue value) { this.value = value; } @Override @Nullable public SkyValue get() { return value; } @Override @Nullable public Exception getException() { return null; } @Override @Nullable public SkyValue getValue() { return value; } @SuppressWarnings("unchecked") private static ValueOrException5ValueImpl ofNullValue() { return (ValueOrException5ValueImpl) NULL; } } private static class ValueOrException5Exn1Impl extends ValueOrException5 { private final E1 e; private ValueOrException5Exn1Impl(E1 e) { this.e = e; } @Override public SkyValue get() throws E1 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException5Exn2Impl extends ValueOrException5 { private final E2 e; private ValueOrException5Exn2Impl(E2 e) { this.e = e; } @Override public SkyValue get() throws E2 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException5Exn3Impl extends ValueOrException5 { private final E3 e; private ValueOrException5Exn3Impl(E3 e) { this.e = e; } @Override public SkyValue get() throws E3 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException5Exn4Impl extends ValueOrException5 { private final E4 e; private ValueOrException5Exn4Impl(E4 e) { this.e = e; } @Override public SkyValue get() throws E4 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } private static class ValueOrException5Exn5Impl extends ValueOrException5 { private final E5 e; private ValueOrException5Exn5Impl(E5 e) { this.e = e; } @Override public SkyValue get() throws E5 { throw e; } @Override public Exception getException() { return e; } @Override @Nullable public SkyValue getValue() { return null; } } }