blob: 0edafcab8c556706fbd5a8af2f8acd9571640f59 [file] [log] [blame]
// Copyright 2022 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.state;
import static com.google.common.base.Preconditions.checkNotNull;
import com.google.devtools.build.skyframe.SkyFunction;
import com.google.devtools.build.skyframe.SkyFunction.LookupEnvironment;
import javax.annotation.Nullable;
/**
* A state machine that outputs a value or one of two possible exceptions.
*
* <p>This class serves as a bridge between a {@link StateMachine} and a {@link SkyFunction}.
*
* <p>Subclasses should call {@link #setValue}, {@link #setException1} or {@link #setException2} to
* emit results.
*/
public abstract class ValueOrException2Producer<V, E1 extends Exception, E2 extends Exception>
implements StateMachine {
private final Driver driver = new Driver(this);
private V value;
private E1 exception1;
private E2 exception2;
/**
* Tries to produce the result of the underlying state machine.
*
* <p>See comment of {@link ValueOrExceptionProducer#tryProduceValue}. The only difference here is
* that if both {@link #exception1} and {@link #exception2} are set, {@link #exception1} has
* priority.
*/
@Nullable
public final V tryProduceValue(LookupEnvironment env) throws InterruptedException, E1, E2 {
boolean done = driver.drive(env);
if (exception1 != null) {
throw exception1;
}
if (exception2 != null) {
throw exception2;
}
if (done) {
return checkNotNull(value);
}
return null;
}
protected final void setValue(V value) {
this.value = value;
}
protected final void setException1(E1 exception) {
this.exception1 = exception;
}
@Nullable
protected final E1 getException1() {
return exception1;
}
protected final void setException2(E2 exception) {
this.exception2 = exception;
}
@Nullable
protected final E2 getException2() {
return exception2;
}
}