| // 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<? extends 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<? extends 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<? extends 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<? extends 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<? extends 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<? extends 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<? extends 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); | 
 |       } | 
 |     }; | 
 |   } | 
 | } |