blob: 5b7847f30b6bc5a65eba17443e2d2b79038b0086 [file] [log] [blame]
/*
* Copyright (C) 2014 The Dagger Authors.
*
* 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 dagger.producers;
import static com.google.common.base.Preconditions.checkNotNull;
import com.google.common.base.Objects;
import com.google.errorprone.annotations.CheckReturnValue;
import dagger.internal.Beta;
import java.util.concurrent.ExecutionException;
import org.checkerframework.checker.nullness.compatqual.NullableDecl;
/**
* An interface that represents the result of a {@linkplain Producer production} of type {@code T},
* or an exception that was thrown during that production. For any type {@code T} that can be
* injected, you can also inject {@code Produced<T>}, which enables handling of any exceptions that
* were thrown during the production of {@code T}.
*
* <p>For example: <pre><code>
* {@literal @}Produces Html getResponse(
* UserInfo criticalInfo, {@literal Produced<ExtraInfo>} noncriticalInfo) {
* try {
* return new Html(criticalInfo, noncriticalInfo.get());
* } catch (ExecutionException e) {
* logger.warning(e, "Noncritical info");
* return new Html(criticalInfo);
* }
* }
* </code></pre>
*
* @since 2.0
*/
@Beta
@CheckReturnValue
public abstract class Produced<T> {
/**
* Returns the result of a production.
*
* @throws ExecutionException if the production threw an exception
*/
public abstract T get() throws ExecutionException;
/**
* Two {@code Produced} objects compare equal if both are successful with equal values, or both
* are failed with equal exceptions.
*/
@Override
public abstract boolean equals(Object o);
/** Returns an appropriate hash code to match {@link #equals(Object)}. */
@Override
public abstract int hashCode();
/** Returns a successful {@code Produced}, whose {@link #get} will return the given value. */
public static <T> Produced<T> successful(@NullableDecl T value) {
return new Successful<T>(value);
}
/**
* Returns a failed {@code Produced}, whose {@link #get} will throw an
* {@code ExecutionException} with the given cause.
*/
public static <T> Produced<T> failed(Throwable throwable) {
return new Failed<T>(checkNotNull(throwable));
}
private static final class Successful<T> extends Produced<T> {
@NullableDecl private final T value;
private Successful(@NullableDecl T value) {
this.value = value;
}
@Override
@NullableDecl
public T get() {
return value;
}
@Override
public boolean equals(Object o) {
if (o == this) {
return true;
} else if (o instanceof Successful) {
Successful<?> that = (Successful<?>) o;
return Objects.equal(this.value, that.value);
} else {
return false;
}
}
@Override
public int hashCode() {
return value == null ? 0 : value.hashCode();
}
@Override
public String toString() {
return "Produced[" + value + "]";
}
}
private static final class Failed<T> extends Produced<T> {
private final Throwable throwable;
private Failed(Throwable throwable) {
this.throwable = checkNotNull(throwable);
}
@Override
public T get() throws ExecutionException {
throw new ExecutionException(throwable);
}
@Override
public boolean equals(Object o) {
if (o == this) {
return true;
} else if (o instanceof Failed) {
Failed<?> that = (Failed<?>) o;
return this.throwable.equals(that.throwable);
} else {
return false;
}
}
@Override
public int hashCode() {
return throwable.hashCode();
}
@Override
public String toString() {
return "Produced[failed with " + throwable.getClass().getCanonicalName() + "]";
}
}
private Produced() {}
}