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