blob: 7b8de1a47cbfaedec149a0b9ad9c1556b236573e [file] [log] [blame]
// Copyright 2015 Google Inc. 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.ImmutableMap;
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 java.util.Set;
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) {
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) {
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) {
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) {
return getValueOrExceptions(ImmutableSet.of(depKey), exceptionClass1, exceptionClass2,
exceptionClass3, exceptionClass4).get(depKey);
}
private <E1 extends Exception, E2 extends Exception, E3 extends Exception,
E4 extends Exception> Map<SkyKey, ValueOrException4<E1, E2, E3, E4>> getValueOrExceptions(
Set<SkyKey> depKeys, Class<E1> exceptionClass1, Class<E2> exceptionClass2,
Class<E3> exceptionClass3, Class<E4> exceptionClass4) {
SkyFunctionException.validateExceptionType(exceptionClass1);
SkyFunctionException.validateExceptionType(exceptionClass2);
SkyFunctionException.validateExceptionType(exceptionClass3);
SkyFunctionException.validateExceptionType(exceptionClass4);
Map<SkyKey, ValueOrUntypedException> valueOrExceptions =
getValueOrUntypedExceptions(depKeys);
ImmutableMap.Builder<SkyKey, ValueOrException4<E1, E2, E3, E4>> builder =
ImmutableMap.builder();
for (SkyKey depKey : depKeys) {
ValueOrUntypedException voe = valueOrExceptions.get(depKey);
SkyValue value = voe.getValue();
if (value != null) {
builder.put(depKey, ValueOrExceptionUtils.<E1, E2, E3, E4>ofValue(value));
continue;
}
Exception e = voe.getException();
if (e != null) {
if (exceptionClass1.isInstance(e)) {
builder.put(depKey, ValueOrExceptionUtils.<E1, E2, E3, E4>ofExn1(
exceptionClass1.cast(e)));
continue;
}
if (exceptionClass2.isInstance(e)) {
builder.put(depKey, ValueOrExceptionUtils.<E1, E2, E3, E4>ofExn2(
exceptionClass2.cast(e)));
continue;
}
if (exceptionClass3.isInstance(e)) {
builder.put(depKey, ValueOrExceptionUtils.<E1, E2, E3, E4>ofExn3(
exceptionClass3.cast(e)));
continue;
}
if (exceptionClass4.isInstance(e)) {
builder.put(depKey, ValueOrExceptionUtils.<E1, E2, E3, E4>ofExn4(
exceptionClass4.cast(e)));
continue;
}
}
valuesMissing = true;
builder.put(depKey, ValueOrExceptionUtils.<E1, E2, E3, E4>ofNullValue());
}
return builder.build();
}
/** Implementations should set {@link #valuesMissing} as necessary. */
protected abstract Map<SkyKey, ValueOrUntypedException> getValueOrUntypedExceptions(
Set<SkyKey> depKeys);
@Override
@Nullable
public SkyValue getValue(SkyKey depKey) {
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 {
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 {
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 {
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 {
return getValueOrException(depKey, exceptionClass1, exceptionClass2, exceptionClass3,
exceptionClass4).get();
}
@Override
public Map<SkyKey, SkyValue> getValues(Iterable<SkyKey> depKeys) {
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) {
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) {
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) {
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) {
Set<SkyKey> keys = ImmutableSet.copyOf(depKeys);
Map<SkyKey, ValueOrException4<E1, E2, E3, E4>> result = getValueOrExceptions(keys,
exceptionClass1, exceptionClass2, exceptionClass3, exceptionClass4);
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);
}
};
}
}