xref: /aosp_15_r20/external/dagger2/java/dagger/producers/internal/SetOfProducedProducer.java (revision f585d8a307d0621d6060bd7e80091fdcbf94fe27)
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