You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
173 lines
7.4 KiB
173 lines
7.4 KiB
/*
|
|
* Copyright (C) 2015 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.internal;
|
|
|
|
import static com.google.common.base.Preconditions.checkNotNull;
|
|
import static com.google.common.util.concurrent.MoreExecutors.directExecutor;
|
|
import static dagger.internal.DaggerCollections.hasDuplicates;
|
|
import static dagger.internal.DaggerCollections.presizedList;
|
|
|
|
import com.google.common.base.Function;
|
|
import com.google.common.collect.ImmutableSet;
|
|
import com.google.common.util.concurrent.Futures;
|
|
import com.google.common.util.concurrent.ListenableFuture;
|
|
import dagger.producers.Produced;
|
|
import dagger.producers.Producer;
|
|
import java.util.ArrayList;
|
|
import java.util.Collection;
|
|
import java.util.List;
|
|
import java.util.Set;
|
|
import java.util.concurrent.ExecutionException;
|
|
|
|
/**
|
|
* A {@link Producer} implementation used to implement {@link Set} bindings. This producer returns a
|
|
* future {@code Set<Produced<T>>} whose elements are populated by subsequent calls to the delegate
|
|
* {@link Producer#get} methods.
|
|
*/
|
|
public final class SetOfProducedProducer<T> extends AbstractProducer<Set<Produced<T>>> {
|
|
public static <T> Producer<Set<T>> empty() {
|
|
return SetProducer.empty();
|
|
}
|
|
|
|
/**
|
|
* Constructs a new {@link Builder} for a {@link SetProducer} with {@code individualProducerSize}
|
|
* individual {@code Producer<T>} and {@code collectionProducerSize} {@code
|
|
* Producer<Collection<T>>} instances.
|
|
*/
|
|
public static <T> Builder<T> builder(int individualProducerSize, int collectionProducerSize) {
|
|
return new Builder<T>(individualProducerSize, collectionProducerSize);
|
|
}
|
|
|
|
/**
|
|
* A builder to accumulate {@code Producer<T>} and {@code Producer<Collection<T>>} instances.
|
|
* These are only intended to be single-use and from within generated code. Do <em>NOT</em> add
|
|
* producers after calling {@link #build()}.
|
|
*/
|
|
public static final class Builder<T> {
|
|
private final List<Producer<T>> individualProducers;
|
|
private final List<Producer<Collection<T>>> collectionProducers;
|
|
|
|
private Builder(int individualProducerSize, int collectionProducerSize) {
|
|
individualProducers = presizedList(individualProducerSize);
|
|
collectionProducers = presizedList(collectionProducerSize);
|
|
}
|
|
|
|
@SuppressWarnings("unchecked")
|
|
public Builder<T> addProducer(Producer<? extends T> individualProducer) {
|
|
assert individualProducer != null : "Codegen error? Null producer";
|
|
individualProducers.add((Producer<T>) individualProducer);
|
|
return this;
|
|
}
|
|
|
|
@SuppressWarnings("unchecked")
|
|
public Builder<T> addCollectionProducer(
|
|
Producer<? extends Collection<? extends T>> multipleProducer) {
|
|
assert multipleProducer != null : "Codegen error? Null producer";
|
|
collectionProducers.add((Producer<Collection<T>>) multipleProducer);
|
|
return this;
|
|
}
|
|
|
|
public SetOfProducedProducer<T> build() {
|
|
assert !hasDuplicates(individualProducers)
|
|
: "Codegen error? Duplicates in the producer list";
|
|
assert !hasDuplicates(collectionProducers)
|
|
: "Codegen error? Duplicates in the producer list";
|
|
|
|
return new SetOfProducedProducer<T>(individualProducers, collectionProducers);
|
|
}
|
|
}
|
|
|
|
private final List<Producer<T>> individualProducers;
|
|
private final List<Producer<Collection<T>>> collectionProducers;
|
|
|
|
private SetOfProducedProducer(
|
|
List<Producer<T>> individualProducers, List<Producer<Collection<T>>> collectionProducers) {
|
|
this.individualProducers = individualProducers;
|
|
this.collectionProducers = collectionProducers;
|
|
}
|
|
|
|
/**
|
|
* Returns a future {@link Set} of {@link Produced} elements given by each of the producers.
|
|
*
|
|
* <p>If any of the delegate collections, or any elements therein, are null, then that
|
|
* corresponding {@code Produced} element will fail with a NullPointerException.
|
|
*
|
|
* <p>Canceling this future will attempt to cancel all of the component futures; but if any of the
|
|
* delegate futures fail or are canceled, this future succeeds, with the appropriate failed {@link
|
|
* Produced}.
|
|
*
|
|
* @throws NullPointerException if any of the delegate producers return null
|
|
*/
|
|
@Override
|
|
public ListenableFuture<Set<Produced<T>>> compute() {
|
|
List<ListenableFuture<? extends Produced<? extends Collection<T>>>> futureProducedCollections =
|
|
new ArrayList<ListenableFuture<? extends Produced<? extends Collection<T>>>>(
|
|
individualProducers.size() + collectionProducers.size());
|
|
for (Producer<T> producer : individualProducers) {
|
|
// TODO(ronshapiro): Don't require individual productions to be added to a collection just to
|
|
// be materialized into futureProducedCollections.
|
|
futureProducedCollections.add(
|
|
Producers.createFutureProduced(
|
|
Producers.createFutureSingletonSet(checkNotNull(producer.get()))));
|
|
}
|
|
for (Producer<Collection<T>> producer : collectionProducers) {
|
|
futureProducedCollections.add(Producers.createFutureProduced(checkNotNull(producer.get())));
|
|
}
|
|
|
|
return Futures.transform(
|
|
Futures.allAsList(futureProducedCollections),
|
|
new Function<List<Produced<? extends Collection<T>>>, Set<Produced<T>>>() {
|
|
@Override
|
|
public Set<Produced<T>> apply(
|
|
List<Produced<? extends Collection<T>>> producedCollections) {
|
|
ImmutableSet.Builder<Produced<T>> builder = ImmutableSet.builder();
|
|
for (Produced<? extends Collection<T>> producedCollection : producedCollections) {
|
|
try {
|
|
Collection<T> collection = producedCollection.get();
|
|
if (collection == null) {
|
|
// TODO(beder): This is a vague exception. Can we somehow point to the failing
|
|
// producer? See the similar comment in the component writer about null
|
|
// provisions.
|
|
builder.add(
|
|
Produced.<T>failed(
|
|
new NullPointerException(
|
|
"Cannot contribute a null collection into a producer set binding when"
|
|
+ " it's injected as Set<Produced<T>>.")));
|
|
} else {
|
|
for (T value : collection) {
|
|
if (value == null) {
|
|
builder.add(
|
|
Produced.<T>failed(
|
|
new NullPointerException(
|
|
"Cannot contribute a null element into a producer set binding"
|
|
+ " when it's injected as Set<Produced<T>>.")));
|
|
} else {
|
|
builder.add(Produced.successful(value));
|
|
}
|
|
}
|
|
}
|
|
} catch (ExecutionException e) {
|
|
builder.add(Produced.<T>failed(e.getCause()));
|
|
}
|
|
}
|
|
return builder.build();
|
|
}
|
|
},
|
|
directExecutor());
|
|
}
|
|
}
|