// Copyright 2015 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 com.google.common.base.Function; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Maps; import com.google.devtools.build.skyframe.ValueOrExceptionUtils.BottomException; import java.util.Collections; import java.util.Map; import javax.annotation.Nullable; /** * Basic implementation of {@link SkyFunction.Environment} in which all convenience methods delegate * to a single abstract method. */ @VisibleForTesting public abstract class AbstractSkyFunctionEnvironment implements SkyFunction.Environment { protected boolean valuesMissing = false; private <E extends Exception> ValueOrException<E> getValueOrException( SkyKey depKey, Class<E> exceptionClass) throws InterruptedException { return ValueOrExceptionUtils.downconvert( getValueOrException(depKey, exceptionClass, BottomException.class), exceptionClass); } private <E1 extends Exception, E2 extends Exception> ValueOrException2<E1, E2> getValueOrException( SkyKey depKey, Class<E1> exceptionClass1, Class<E2> exceptionClass2) throws InterruptedException { return ValueOrExceptionUtils.downconvert(getValueOrException(depKey, exceptionClass1, exceptionClass2, BottomException.class), exceptionClass1, exceptionClass2); } private <E1 extends Exception, E2 extends Exception, E3 extends Exception> ValueOrException3<E1, E2, E3> getValueOrException( SkyKey depKey, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3) throws InterruptedException { return ValueOrExceptionUtils.downconvert( getValueOrException(depKey, exceptionClass1, exceptionClass2, exceptionClass3, BottomException.class), exceptionClass1, exceptionClass2, exceptionClass3); } private <E1 extends Exception, E2 extends Exception, E3 extends Exception, E4 extends Exception> ValueOrException4<E1, E2, E3, E4> getValueOrException( SkyKey depKey, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3, Class<E4> exceptionClass4) throws InterruptedException { return ValueOrExceptionUtils.downconvert( getValueOrException(depKey, exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4, BottomException.class), exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4); } private < E1 extends Exception, E2 extends Exception, E3 extends Exception, E4 extends Exception, E5 extends Exception> ValueOrException5<E1, E2, E3, E4, E5> getValueOrException( SkyKey depKey, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3, Class<E4> exceptionClass4, Class<E5> exceptionClass5) throws InterruptedException { return getValueOrExceptions( ImmutableSet.of(depKey), exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4, exceptionClass5).get(depKey); } private < E1 extends Exception, E2 extends Exception, E3 extends Exception, E4 extends Exception, E5 extends Exception> Map<SkyKey, ValueOrException5<E1, E2, E3, E4, E5>> getValueOrExceptions( Iterable<SkyKey> depKeys, final Class<E1> exceptionClass1, final Class<E2> exceptionClass2, final Class<E3> exceptionClass3, final Class<E4> exceptionClass4, final Class<E5> exceptionClass5) throws InterruptedException { SkyFunctionException.validateExceptionType(exceptionClass1); SkyFunctionException.validateExceptionType(exceptionClass2); SkyFunctionException.validateExceptionType(exceptionClass3); SkyFunctionException.validateExceptionType(exceptionClass4); SkyFunctionException.validateExceptionType(exceptionClass5); Map<SkyKey, ValueOrUntypedException> valueOrExceptions = getValueOrUntypedExceptions(depKeys); for (ValueOrUntypedException voe : valueOrExceptions.values()) { SkyValue value = voe.getValue(); if (value == null) { Exception e = voe.getException(); if (e == null || (!exceptionClass1.isInstance(e) && !exceptionClass2.isInstance(e) && !exceptionClass3.isInstance(e) && !exceptionClass4.isInstance(e) && !exceptionClass5.isInstance(e))) { valuesMissing = true; break; } } } // We transform the values directly to avoid the transient memory cost of creating a new map. return Maps.transformValues( valueOrExceptions, new Function<ValueOrUntypedException, ValueOrException5<E1, E2, E3, E4, E5>>() { @Override public ValueOrException5<E1, E2, E3, E4, E5> apply(ValueOrUntypedException voe) { SkyValue value = voe.getValue(); if (value != null) { return ValueOrExceptionUtils.<E1, E2, E3, E4, E5>ofValue(value); } Exception e = voe.getException(); if (e != null) { if (exceptionClass1.isInstance(e)) { return ValueOrExceptionUtils.<E1, E2, E3, E4, E5>ofExn1(exceptionClass1.cast(e)); } if (exceptionClass2.isInstance(e)) { return ValueOrExceptionUtils.<E1, E2, E3, E4, E5>ofExn2(exceptionClass2.cast(e)); } if (exceptionClass3.isInstance(e)) { return ValueOrExceptionUtils.<E1, E2, E3, E4, E5>ofExn3(exceptionClass3.cast(e)); } if (exceptionClass4.isInstance(e)) { return ValueOrExceptionUtils.<E1, E2, E3, E4, E5>ofExn4(exceptionClass4.cast(e)); } if (exceptionClass5.isInstance(e)) { return ValueOrExceptionUtils.<E1, E2, E3, E4, E5>ofExn5(exceptionClass5.cast(e)); } } return ValueOrExceptionUtils.<E1, E2, E3, E4, E5>ofNullValue(); } }); } /** Implementations should set {@link #valuesMissing} as necessary. */ protected abstract Map<SkyKey, ValueOrUntypedException> getValueOrUntypedExceptions( Iterable<SkyKey> depKeys) throws InterruptedException; @Override @Nullable public SkyValue getValue(SkyKey depKey) throws InterruptedException { try { return getValueOrThrow(depKey, BottomException.class); } catch (BottomException e) { throw new IllegalStateException("shouldn't reach here"); } } @Override @Nullable public <E extends Exception> SkyValue getValueOrThrow(SkyKey depKey, Class<E> exceptionClass) throws E, InterruptedException { return getValueOrException(depKey, exceptionClass).get(); } @Override @Nullable public <E1 extends Exception, E2 extends Exception> SkyValue getValueOrThrow( SkyKey depKey, Class<E1> exceptionClass1, Class<E2> exceptionClass2) throws E1, E2, InterruptedException { return getValueOrException(depKey, exceptionClass1, exceptionClass2).get(); } @Override @Nullable public <E1 extends Exception, E2 extends Exception, E3 extends Exception> SkyValue getValueOrThrow( SkyKey depKey, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3) throws E1, E2, E3, InterruptedException { return getValueOrException(depKey, exceptionClass1, exceptionClass2, exceptionClass3).get(); } @Override public <E1 extends Exception, E2 extends Exception, E3 extends Exception, E4 extends Exception> SkyValue getValueOrThrow( SkyKey depKey, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3, Class<E4> exceptionClass4) throws E1, E2, E3, E4, InterruptedException { return getValueOrException( depKey, exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4).get(); } @Override public < E1 extends Exception, E2 extends Exception, E3 extends Exception, E4 extends Exception, E5 extends Exception> SkyValue getValueOrThrow( SkyKey depKey, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3, Class<E4> exceptionClass4, Class<E5> exceptionClass5) throws E1, E2, E3, E4, E5, InterruptedException { return getValueOrException( depKey, exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4, exceptionClass5).get(); } @Override public Map<SkyKey, SkyValue> getValues(Iterable<SkyKey> depKeys) throws InterruptedException { return Maps.transformValues(getValuesOrThrow(depKeys, BottomException.class), GET_VALUE_FROM_VOE); } @Override public <E extends Exception> Map<SkyKey, ValueOrException<E>> getValuesOrThrow( Iterable<SkyKey> depKeys, Class<E> exceptionClass) throws InterruptedException { return Maps.transformValues( getValuesOrThrow(depKeys, exceptionClass, BottomException.class), makeSafeDowncastToVOEFunction(exceptionClass)); } @Override public <E1 extends Exception, E2 extends Exception> Map<SkyKey, ValueOrException2<E1, E2>> getValuesOrThrow( Iterable<SkyKey> depKeys, Class<E1> exceptionClass1, Class<E2> exceptionClass2) throws InterruptedException { return Maps.transformValues( getValuesOrThrow(depKeys, exceptionClass1, exceptionClass2, BottomException.class), makeSafeDowncastToVOE2Function(exceptionClass1, exceptionClass2)); } @Override public <E1 extends Exception, E2 extends Exception, E3 extends Exception> Map<SkyKey, ValueOrException3<E1, E2, E3>> getValuesOrThrow( Iterable<SkyKey> depKeys, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3) throws InterruptedException { return Maps.transformValues( getValuesOrThrow(depKeys, exceptionClass1, exceptionClass2, exceptionClass3, BottomException.class), makeSafeDowncastToVOE3Function(exceptionClass1, exceptionClass2, exceptionClass3)); } @Override public <E1 extends Exception, E2 extends Exception, E3 extends Exception, E4 extends Exception> Map<SkyKey, ValueOrException4<E1, E2, E3, E4>> getValuesOrThrow( Iterable<SkyKey> depKeys, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3, Class<E4> exceptionClass4) throws InterruptedException { return Maps.transformValues( getValuesOrThrow(depKeys, exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4, BottomException.class), makeSafeDowncastToVOE4Function(exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4)); } @Override public < E1 extends Exception, E2 extends Exception, E3 extends Exception, E4 extends Exception, E5 extends Exception> Map<SkyKey, ValueOrException5<E1, E2, E3, E4, E5>> getValuesOrThrow( Iterable<SkyKey> depKeys, Class<E1> exceptionClass1, Class<E2> exceptionClass2, Class<E3> exceptionClass3, Class<E4> exceptionClass4, Class<E5> exceptionClass5) throws InterruptedException { Map<SkyKey, ValueOrException5<E1, E2, E3, E4, E5>> result = getValueOrExceptions( depKeys, exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4, exceptionClass5); return Collections.unmodifiableMap(result); } @Override public boolean valuesMissing() { return valuesMissing; } private static final Function<ValueOrException<BottomException>, SkyValue> GET_VALUE_FROM_VOE = new Function<ValueOrException<BottomException>, SkyValue>() { @Override public SkyValue apply(ValueOrException<BottomException> voe) { return ValueOrExceptionUtils.downconvert(voe); } }; private static <E extends Exception> Function<ValueOrException2<E, BottomException>, ValueOrException<E>> makeSafeDowncastToVOEFunction(final Class<E> exceptionClass) { return new Function<ValueOrException2<E, BottomException>, ValueOrException<E>>() { @Override public ValueOrException<E> apply(ValueOrException2<E, BottomException> voe) { return ValueOrExceptionUtils.downconvert(voe, exceptionClass); } }; } private static <E1 extends Exception, E2 extends Exception> Function<ValueOrException3<E1, E2, BottomException>, ValueOrException2<E1, E2>> makeSafeDowncastToVOE2Function( final Class<E1> exceptionClass1, final Class<E2> exceptionClass2) { return new Function<ValueOrException3<E1, E2, BottomException>, ValueOrException2<E1, E2>>() { @Override public ValueOrException2<E1, E2> apply(ValueOrException3<E1, E2, BottomException> voe) { return ValueOrExceptionUtils.downconvert(voe, exceptionClass1, exceptionClass2); } }; } private static <E1 extends Exception, E2 extends Exception, E3 extends Exception> Function<ValueOrException4<E1, E2, E3, BottomException>, ValueOrException3<E1, E2, E3>> makeSafeDowncastToVOE3Function( final Class<E1> exceptionClass1, final Class<E2> exceptionClass2, final Class<E3> exceptionClass3) { return new Function<ValueOrException4<E1, E2, E3, BottomException>, ValueOrException3<E1, E2, E3>>() { @Override public ValueOrException3<E1, E2, E3> apply( ValueOrException4<E1, E2, E3, BottomException> voe) { return ValueOrExceptionUtils.downconvert( voe, exceptionClass1, exceptionClass2, exceptionClass3); } }; } private static <E1 extends Exception, E2 extends Exception, E3 extends Exception, E4 extends Exception> Function<ValueOrException5<E1, E2, E3, E4, BottomException>, ValueOrException4<E1, E2, E3, E4>> makeSafeDowncastToVOE4Function( final Class<E1> exceptionClass1, final Class<E2> exceptionClass2, final Class<E3> exceptionClass3, final Class<E4> exceptionClass4) { return new Function<ValueOrException5<E1, E2, E3, E4, BottomException>, ValueOrException4<E1, E2, E3, E4>>() { @Override public ValueOrException4<E1, E2, E3, E4> apply( ValueOrException5<E1, E2, E3, E4, BottomException> voe) { return ValueOrExceptionUtils.downconvert( voe, exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4); } }; } }