xref: /aosp_15_r20/external/dagger2/java/dagger/producers/internal/MapOfProducedProducer.java (revision f585d8a307d0621d6060bd7e80091fdcbf94fe27)
1*f585d8a3SJacky Wang /*
2*f585d8a3SJacky Wang  * Copyright (C) 2016 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.util.concurrent.Futures.transform;
20*f585d8a3SJacky Wang import static com.google.common.util.concurrent.MoreExecutors.directExecutor;
21*f585d8a3SJacky Wang import static dagger.internal.Providers.asDaggerProvider;
22*f585d8a3SJacky Wang 
23*f585d8a3SJacky Wang import com.google.common.base.Function;
24*f585d8a3SJacky Wang import com.google.common.collect.ImmutableMap;
25*f585d8a3SJacky Wang import com.google.common.collect.Iterables;
26*f585d8a3SJacky Wang import com.google.common.collect.Maps;
27*f585d8a3SJacky Wang import com.google.common.util.concurrent.Futures;
28*f585d8a3SJacky Wang import com.google.common.util.concurrent.ListenableFuture;
29*f585d8a3SJacky Wang import dagger.internal.Provider;
30*f585d8a3SJacky Wang import dagger.producers.Produced;
31*f585d8a3SJacky Wang import dagger.producers.Producer;
32*f585d8a3SJacky Wang import java.util.List;
33*f585d8a3SJacky Wang import java.util.Map;
34*f585d8a3SJacky Wang 
35*f585d8a3SJacky Wang /**
36*f585d8a3SJacky Wang  * A {@link Producer} implementation used to implement {@link Map} bindings. This producer returns a
37*f585d8a3SJacky Wang  * {@code Map<K, Produced<V>>} which is populated by calls to the delegate {@link Producer#get}
38*f585d8a3SJacky Wang  * methods.
39*f585d8a3SJacky Wang  */
40*f585d8a3SJacky Wang public final class MapOfProducedProducer<K, V> extends AbstractMapProducer<K, V, Produced<V>> {
MapOfProducedProducer(ImmutableMap<K, Producer<V>> contributingMap)41*f585d8a3SJacky Wang   private MapOfProducedProducer(ImmutableMap<K, Producer<V>> contributingMap) {
42*f585d8a3SJacky Wang     super(contributingMap);
43*f585d8a3SJacky Wang   }
44*f585d8a3SJacky Wang 
45*f585d8a3SJacky Wang   @Override
compute()46*f585d8a3SJacky Wang   public ListenableFuture<Map<K, Produced<V>>> compute() {
47*f585d8a3SJacky Wang     return Futures.transform(
48*f585d8a3SJacky Wang         Futures.allAsList(
49*f585d8a3SJacky Wang             Iterables.transform(
50*f585d8a3SJacky Wang                 contributingMap().entrySet(), MapOfProducedProducer.<K, V>entryUnwrapper())),
51*f585d8a3SJacky Wang         new Function<List<Map.Entry<K, Produced<V>>>, Map<K, Produced<V>>>() {
52*f585d8a3SJacky Wang           @Override
53*f585d8a3SJacky Wang           public Map<K, Produced<V>> apply(List<Map.Entry<K, Produced<V>>> entries) {
54*f585d8a3SJacky Wang             return ImmutableMap.copyOf(entries);
55*f585d8a3SJacky Wang           }
56*f585d8a3SJacky Wang         },
57*f585d8a3SJacky Wang         directExecutor());
58*f585d8a3SJacky Wang   }
59*f585d8a3SJacky Wang 
60*f585d8a3SJacky Wang   private static final Function<
61*f585d8a3SJacky Wang           Map.Entry<Object, Producer<Object>>,
62*f585d8a3SJacky Wang           ListenableFuture<Map.Entry<Object, Produced<Object>>>>
63*f585d8a3SJacky Wang       ENTRY_UNWRAPPER =
64*f585d8a3SJacky Wang           new Function<
65*f585d8a3SJacky Wang               Map.Entry<Object, Producer<Object>>,
66*f585d8a3SJacky Wang               ListenableFuture<Map.Entry<Object, Produced<Object>>>>() {
67*f585d8a3SJacky Wang             @Override
68*f585d8a3SJacky Wang             public ListenableFuture<Map.Entry<Object, Produced<Object>>> apply(
69*f585d8a3SJacky Wang                 final Map.Entry<Object, Producer<Object>> entry) {
70*f585d8a3SJacky Wang               return transform(
71*f585d8a3SJacky Wang                   Producers.createFutureProduced(entry.getValue().get()),
72*f585d8a3SJacky Wang                   new Function<Produced<Object>, Map.Entry<Object, Produced<Object>>>() {
73*f585d8a3SJacky Wang                     @Override
74*f585d8a3SJacky Wang                     public Map.Entry<Object, Produced<Object>> apply(Produced<Object> value) {
75*f585d8a3SJacky Wang                       return Maps.immutableEntry(entry.getKey(), value);
76*f585d8a3SJacky Wang                     }
77*f585d8a3SJacky Wang                   },
78*f585d8a3SJacky Wang                   directExecutor());
79*f585d8a3SJacky Wang             }
80*f585d8a3SJacky Wang           };
81*f585d8a3SJacky Wang 
82*f585d8a3SJacky Wang   @SuppressWarnings({"unchecked", "rawtypes"}) // bivariate implementation
83*f585d8a3SJacky Wang   private static <K, V>
84*f585d8a3SJacky Wang       Function<Map.Entry<K, Producer<V>>, ListenableFuture<Map.Entry<K, Produced<V>>>>
85*f585d8a3SJacky Wang           entryUnwrapper() {
86*f585d8a3SJacky Wang     return (Function) ENTRY_UNWRAPPER;
87*f585d8a3SJacky Wang   }
88*f585d8a3SJacky Wang 
89*f585d8a3SJacky Wang   /** Returns a new {@link Builder}. */
90*f585d8a3SJacky Wang   public static <K, V> Builder<K, V> builder(int size) {
91*f585d8a3SJacky Wang     return new Builder<>(size);
92*f585d8a3SJacky Wang   }
93*f585d8a3SJacky Wang 
94*f585d8a3SJacky Wang   /** A builder for {@link MapOfProducedProducer}. */
95*f585d8a3SJacky Wang   public static final class Builder<K, V> extends AbstractMapProducer.Builder<K, V, Produced<V>> {
96*f585d8a3SJacky Wang     private Builder(int size) {
97*f585d8a3SJacky Wang       super(size);
98*f585d8a3SJacky Wang     }
99*f585d8a3SJacky Wang 
100*f585d8a3SJacky Wang     @Override
101*f585d8a3SJacky Wang     public Builder<K, V> put(K key, Producer<V> producerOfValue) {
102*f585d8a3SJacky Wang       super.put(key, producerOfValue);
103*f585d8a3SJacky Wang       return this;
104*f585d8a3SJacky Wang     }
105*f585d8a3SJacky Wang 
106*f585d8a3SJacky Wang     @Override
107*f585d8a3SJacky Wang     public Builder<K, V> put(K key, Provider<V> providerOfValue) {
108*f585d8a3SJacky Wang       super.put(key, providerOfValue);
109*f585d8a3SJacky Wang       return this;
110*f585d8a3SJacky Wang     }
111*f585d8a3SJacky Wang 
112*f585d8a3SJacky Wang     /**
113*f585d8a3SJacky Wang      * Legacy javax version of the method to support libraries compiled with an older version of
114*f585d8a3SJacky Wang      * Dagger. Do not use directly.
115*f585d8a3SJacky Wang      */
116*f585d8a3SJacky Wang     @Deprecated
117*f585d8a3SJacky Wang     public Builder<K, V> put(K key, javax.inject.Provider<V> providerOfValue) {
118*f585d8a3SJacky Wang       return put(key, asDaggerProvider(providerOfValue));
119*f585d8a3SJacky Wang     }
120*f585d8a3SJacky Wang 
121*f585d8a3SJacky Wang     @Override
122*f585d8a3SJacky Wang     public Builder<K, V> putAll(Producer<Map<K, Produced<V>>> mapOfProducedProducer) {
123*f585d8a3SJacky Wang       super.putAll(mapOfProducedProducer);
124*f585d8a3SJacky Wang       return this;
125*f585d8a3SJacky Wang     }
126*f585d8a3SJacky Wang 
127*f585d8a3SJacky Wang     /** Returns a new {@link MapOfProducedProducer}. */
128*f585d8a3SJacky Wang     public MapOfProducedProducer<K, V> build() {
129*f585d8a3SJacky Wang       return new MapOfProducedProducer<>(mapBuilder.build());
130*f585d8a3SJacky Wang     }
131*f585d8a3SJacky Wang   }
132*f585d8a3SJacky Wang }
133