1*f585d8a3SJacky Wang /* 2*f585d8a3SJacky Wang * Copyright (C) 2015 The Dagger Authors. 3*f585d8a3SJacky Wang * 4*f585d8a3SJacky Wang * Licensed under the Apache License, Version 2.0 (the "License"); 5*f585d8a3SJacky Wang * you may not use this file except in compliance with the License. 6*f585d8a3SJacky Wang * You may obtain a copy of the License at 7*f585d8a3SJacky Wang * 8*f585d8a3SJacky Wang * http://www.apache.org/licenses/LICENSE-2.0 9*f585d8a3SJacky Wang * 10*f585d8a3SJacky Wang * Unless required by applicable law or agreed to in writing, software 11*f585d8a3SJacky Wang * distributed under the License is distributed on an "AS IS" BASIS, 12*f585d8a3SJacky Wang * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*f585d8a3SJacky Wang * See the License for the specific language governing permissions and 14*f585d8a3SJacky Wang * limitations under the License. 15*f585d8a3SJacky Wang */ 16*f585d8a3SJacky Wang 17*f585d8a3SJacky Wang package dagger.producers.internal; 18*f585d8a3SJacky Wang 19*f585d8a3SJacky Wang import static com.google.common.base.Preconditions.checkNotNull; 20*f585d8a3SJacky Wang import static com.google.common.util.concurrent.MoreExecutors.directExecutor; 21*f585d8a3SJacky Wang import static dagger.internal.DaggerCollections.hasDuplicates; 22*f585d8a3SJacky Wang import static dagger.internal.DaggerCollections.presizedList; 23*f585d8a3SJacky Wang 24*f585d8a3SJacky Wang import com.google.common.base.Function; 25*f585d8a3SJacky Wang import com.google.common.collect.ImmutableSet; 26*f585d8a3SJacky Wang import com.google.common.util.concurrent.Futures; 27*f585d8a3SJacky Wang import com.google.common.util.concurrent.ListenableFuture; 28*f585d8a3SJacky Wang import dagger.producers.Produced; 29*f585d8a3SJacky Wang import dagger.producers.Producer; 30*f585d8a3SJacky Wang import java.util.ArrayList; 31*f585d8a3SJacky Wang import java.util.Collection; 32*f585d8a3SJacky Wang import java.util.List; 33*f585d8a3SJacky Wang import java.util.Set; 34*f585d8a3SJacky Wang import java.util.concurrent.ExecutionException; 35*f585d8a3SJacky Wang 36*f585d8a3SJacky Wang /** 37*f585d8a3SJacky Wang * A {@link Producer} implementation used to implement {@link Set} bindings. This producer returns a 38*f585d8a3SJacky Wang * future {@code Set<Produced<T>>} whose elements are populated by subsequent calls to the delegate 39*f585d8a3SJacky Wang * {@link Producer#get} methods. 40*f585d8a3SJacky Wang */ 41*f585d8a3SJacky Wang public final class SetOfProducedProducer<T> extends AbstractProducer<Set<Produced<T>>> { empty()42*f585d8a3SJacky Wang public static <T> Producer<Set<T>> empty() { 43*f585d8a3SJacky Wang return SetProducer.empty(); 44*f585d8a3SJacky Wang } 45*f585d8a3SJacky Wang 46*f585d8a3SJacky Wang /** 47*f585d8a3SJacky Wang * Constructs a new {@link Builder} for a {@link SetProducer} with {@code individualProducerSize} 48*f585d8a3SJacky Wang * individual {@code Producer<T>} and {@code collectionProducerSize} {@code 49*f585d8a3SJacky Wang * Producer<Collection<T>>} instances. 50*f585d8a3SJacky Wang */ builder(int individualProducerSize, int collectionProducerSize)51*f585d8a3SJacky Wang public static <T> Builder<T> builder(int individualProducerSize, int collectionProducerSize) { 52*f585d8a3SJacky Wang return new Builder<T>(individualProducerSize, collectionProducerSize); 53*f585d8a3SJacky Wang } 54*f585d8a3SJacky Wang 55*f585d8a3SJacky Wang /** 56*f585d8a3SJacky Wang * A builder to accumulate {@code Producer<T>} and {@code Producer<Collection<T>>} instances. 57*f585d8a3SJacky Wang * These are only intended to be single-use and from within generated code. Do <em>NOT</em> add 58*f585d8a3SJacky Wang * producers after calling {@link #build()}. 59*f585d8a3SJacky Wang */ 60*f585d8a3SJacky Wang public static final class Builder<T> { 61*f585d8a3SJacky Wang private final List<Producer<T>> individualProducers; 62*f585d8a3SJacky Wang private final List<Producer<Collection<T>>> collectionProducers; 63*f585d8a3SJacky Wang Builder(int individualProducerSize, int collectionProducerSize)64*f585d8a3SJacky Wang private Builder(int individualProducerSize, int collectionProducerSize) { 65*f585d8a3SJacky Wang individualProducers = presizedList(individualProducerSize); 66*f585d8a3SJacky Wang collectionProducers = presizedList(collectionProducerSize); 67*f585d8a3SJacky Wang } 68*f585d8a3SJacky Wang 69*f585d8a3SJacky Wang @SuppressWarnings("unchecked") addProducer(Producer<? extends T> individualProducer)70*f585d8a3SJacky Wang public Builder<T> addProducer(Producer<? extends T> individualProducer) { 71*f585d8a3SJacky Wang assert individualProducer != null : "Codegen error? Null producer"; 72*f585d8a3SJacky Wang individualProducers.add((Producer<T>) individualProducer); 73*f585d8a3SJacky Wang return this; 74*f585d8a3SJacky Wang } 75*f585d8a3SJacky Wang 76*f585d8a3SJacky Wang @SuppressWarnings("unchecked") addCollectionProducer( Producer<? extends Collection<? extends T>> multipleProducer)77*f585d8a3SJacky Wang public Builder<T> addCollectionProducer( 78*f585d8a3SJacky Wang Producer<? extends Collection<? extends T>> multipleProducer) { 79*f585d8a3SJacky Wang assert multipleProducer != null : "Codegen error? Null producer"; 80*f585d8a3SJacky Wang collectionProducers.add((Producer<Collection<T>>) multipleProducer); 81*f585d8a3SJacky Wang return this; 82*f585d8a3SJacky Wang } 83*f585d8a3SJacky Wang build()84*f585d8a3SJacky Wang public SetOfProducedProducer<T> build() { 85*f585d8a3SJacky Wang assert !hasDuplicates(individualProducers) 86*f585d8a3SJacky Wang : "Codegen error? Duplicates in the producer list"; 87*f585d8a3SJacky Wang assert !hasDuplicates(collectionProducers) 88*f585d8a3SJacky Wang : "Codegen error? Duplicates in the producer list"; 89*f585d8a3SJacky Wang 90*f585d8a3SJacky Wang return new SetOfProducedProducer<T>(individualProducers, collectionProducers); 91*f585d8a3SJacky Wang } 92*f585d8a3SJacky Wang } 93*f585d8a3SJacky Wang 94*f585d8a3SJacky Wang private final List<Producer<T>> individualProducers; 95*f585d8a3SJacky Wang private final List<Producer<Collection<T>>> collectionProducers; 96*f585d8a3SJacky Wang SetOfProducedProducer( List<Producer<T>> individualProducers, List<Producer<Collection<T>>> collectionProducers)97*f585d8a3SJacky Wang private SetOfProducedProducer( 98*f585d8a3SJacky Wang List<Producer<T>> individualProducers, List<Producer<Collection<T>>> collectionProducers) { 99*f585d8a3SJacky Wang this.individualProducers = individualProducers; 100*f585d8a3SJacky Wang this.collectionProducers = collectionProducers; 101*f585d8a3SJacky Wang } 102*f585d8a3SJacky Wang 103*f585d8a3SJacky Wang /** 104*f585d8a3SJacky Wang * Returns a future {@link Set} of {@link Produced} elements given by each of the producers. 105*f585d8a3SJacky Wang * 106*f585d8a3SJacky Wang * <p>If any of the delegate collections, or any elements therein, are null, then that 107*f585d8a3SJacky Wang * corresponding {@code Produced} element will fail with a NullPointerException. 108*f585d8a3SJacky Wang * 109*f585d8a3SJacky Wang * <p>Canceling this future will attempt to cancel all of the component futures; but if any of the 110*f585d8a3SJacky Wang * delegate futures fail or are canceled, this future succeeds, with the appropriate failed {@link 111*f585d8a3SJacky Wang * Produced}. 112*f585d8a3SJacky Wang * 113*f585d8a3SJacky Wang * @throws NullPointerException if any of the delegate producers return null 114*f585d8a3SJacky Wang */ 115*f585d8a3SJacky Wang @Override compute()116*f585d8a3SJacky Wang public ListenableFuture<Set<Produced<T>>> compute() { 117*f585d8a3SJacky Wang List<ListenableFuture<? extends Produced<? extends Collection<T>>>> futureProducedCollections = 118*f585d8a3SJacky Wang new ArrayList<ListenableFuture<? extends Produced<? extends Collection<T>>>>( 119*f585d8a3SJacky Wang individualProducers.size() + collectionProducers.size()); 120*f585d8a3SJacky Wang for (Producer<T> producer : individualProducers) { 121*f585d8a3SJacky Wang // TODO(ronshapiro): Don't require individual productions to be added to a collection just to 122*f585d8a3SJacky Wang // be materialized into futureProducedCollections. 123*f585d8a3SJacky Wang futureProducedCollections.add( 124*f585d8a3SJacky Wang Producers.createFutureProduced( 125*f585d8a3SJacky Wang Producers.createFutureSingletonSet(checkNotNull(producer.get())))); 126*f585d8a3SJacky Wang } 127*f585d8a3SJacky Wang for (Producer<Collection<T>> producer : collectionProducers) { 128*f585d8a3SJacky Wang futureProducedCollections.add(Producers.createFutureProduced(checkNotNull(producer.get()))); 129*f585d8a3SJacky Wang } 130*f585d8a3SJacky Wang 131*f585d8a3SJacky Wang return Futures.transform( 132*f585d8a3SJacky Wang Futures.allAsList(futureProducedCollections), 133*f585d8a3SJacky Wang new Function<List<Produced<? extends Collection<T>>>, Set<Produced<T>>>() { 134*f585d8a3SJacky Wang @Override 135*f585d8a3SJacky Wang public Set<Produced<T>> apply( 136*f585d8a3SJacky Wang List<Produced<? extends Collection<T>>> producedCollections) { 137*f585d8a3SJacky Wang ImmutableSet.Builder<Produced<T>> builder = ImmutableSet.builder(); 138*f585d8a3SJacky Wang for (Produced<? extends Collection<T>> producedCollection : producedCollections) { 139*f585d8a3SJacky Wang try { 140*f585d8a3SJacky Wang Collection<T> collection = producedCollection.get(); 141*f585d8a3SJacky Wang if (collection == null) { 142*f585d8a3SJacky Wang // TODO(beder): This is a vague exception. Can we somehow point to the failing 143*f585d8a3SJacky Wang // producer? See the similar comment in the component writer about null 144*f585d8a3SJacky Wang // provisions. 145*f585d8a3SJacky Wang builder.add( 146*f585d8a3SJacky Wang Produced.<T>failed( 147*f585d8a3SJacky Wang new NullPointerException( 148*f585d8a3SJacky Wang "Cannot contribute a null collection into a producer set binding when" 149*f585d8a3SJacky Wang + " it's injected as Set<Produced<T>>."))); 150*f585d8a3SJacky Wang } else { 151*f585d8a3SJacky Wang for (T value : collection) { 152*f585d8a3SJacky Wang if (value == null) { 153*f585d8a3SJacky Wang builder.add( 154*f585d8a3SJacky Wang Produced.<T>failed( 155*f585d8a3SJacky Wang new NullPointerException( 156*f585d8a3SJacky Wang "Cannot contribute a null element into a producer set binding" 157*f585d8a3SJacky Wang + " when it's injected as Set<Produced<T>>."))); 158*f585d8a3SJacky Wang } else { 159*f585d8a3SJacky Wang builder.add(Produced.successful(value)); 160*f585d8a3SJacky Wang } 161*f585d8a3SJacky Wang } 162*f585d8a3SJacky Wang } 163*f585d8a3SJacky Wang } catch (ExecutionException e) { 164*f585d8a3SJacky Wang builder.add(Produced.<T>failed(e.getCause())); 165*f585d8a3SJacky Wang } 166*f585d8a3SJacky Wang } 167*f585d8a3SJacky Wang return builder.build(); 168*f585d8a3SJacky Wang } 169*f585d8a3SJacky Wang }, 170*f585d8a3SJacky Wang directExecutor()); 171*f585d8a3SJacky Wang } 172*f585d8a3SJacky Wang } 173