/* * Copyright (C) 2023 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.functional.kotlinsrc.subcomponent.multibindings import com.google.common.collect.ImmutableMap import com.google.common.collect.ImmutableSet import com.google.common.truth.Truth.assertThat import dagger.functional.kotlinsrc.subcomponent.multibindings.MultibindingSubcomponents.* import org.junit.Before import org.junit.Test import org.junit.runner.RunWith import org.junit.runners.JUnit4 @RunWith(JUnit4::class) class SubcomponentMultibindingsTest { private lateinit var parentWithoutProvisionHasChildWithoutProvision: ParentWithoutProvisionHasChildWithoutProvision private lateinit var parentWithoutProvisionHasChildWithProvision: ParentWithoutProvisionHasChildWithProvision private lateinit var parentWithProvisionHasChildWithoutProvision: ParentWithProvisionHasChildWithoutProvision private lateinit var parentWithProvisionHasChildWithProvision: ParentWithProvisionHasChildWithProvision @Before fun setUp() { parentWithoutProvisionHasChildWithoutProvision = DaggerMultibindingSubcomponents_ParentWithoutProvisionHasChildWithoutProvision.create() parentWithoutProvisionHasChildWithProvision = DaggerMultibindingSubcomponents_ParentWithoutProvisionHasChildWithProvision.create() parentWithProvisionHasChildWithoutProvision = DaggerMultibindingSubcomponents_ParentWithProvisionHasChildWithoutProvision.create() parentWithProvisionHasChildWithProvision = DaggerMultibindingSubcomponents_ParentWithProvisionHasChildWithProvision.create() } @Test fun testParentWithoutProvisionHasChildWithoutProvision() { // Child assertThat( parentWithoutProvisionHasChildWithoutProvision .childWithoutProvision() .grandchild() .requiresMultibindingsBoundInParent() ) .isEqualTo(BOUND_IN_PARENT) // Grandchild assertThat( parentWithoutProvisionHasChildWithoutProvision .childWithoutProvision() .grandchild() .requiresMultibindingsBoundInParentAndChild() ) .isEqualTo(BOUND_IN_PARENT_AND_CHILD) assertThat( parentWithoutProvisionHasChildWithoutProvision .childWithoutProvision() .grandchild() .requiresMultibindingsBoundInChild() ) .isEqualTo(BOUND_IN_CHILD) // Even though the multibinding for Set does not itself have a // contribution from the child, it must be pushed down to (not duplicated in) the child because // its contribution depends on multibindings that have one contribution from the parent and one // from the child (see b/22821341). assertThat( parentWithoutProvisionHasChildWithoutProvision .childWithoutProvision() .grandchild() .setOfRequiresMultibindingsInParentAndChild() ) .containsExactly(BOUND_IN_PARENT_AND_CHILD) } @Test fun testParentWithoutProvisionHasChildWithProvision() { // Child assertThat( parentWithoutProvisionHasChildWithProvision .childWithProvision() .grandchild() .requiresMultibindingsBoundInParent() ) .isEqualTo(BOUND_IN_PARENT) // Grandchild assertThat( parentWithoutProvisionHasChildWithProvision .childWithProvision() .grandchild() .requiresMultibindingsBoundInParentAndChild() ) .isEqualTo(BOUND_IN_PARENT_AND_CHILD) assertThat( parentWithoutProvisionHasChildWithProvision .childWithProvision() .grandchild() .requiresMultibindingsBoundInChild() ) .isEqualTo(BOUND_IN_CHILD) // Even though the multibinding for Set does not itself have a // contribution from the child, it must be pushed down to (not duplicated in) the child because // its contribution depends on multibindings that have one contribution from the parent and one // from the child (see b/22821341). assertThat( parentWithoutProvisionHasChildWithProvision .childWithProvision() .grandchild() .setOfRequiresMultibindingsInParentAndChild() ) .containsExactly(BOUND_IN_PARENT_AND_CHILD) } @Test fun testParentWithProvisionHasChildWithoutProvision() { // Parent assertThat(parentWithProvisionHasChildWithoutProvision.requiresMultibindingsBoundInParent()) .isEqualTo(BOUND_IN_PARENT) assertThat( parentWithProvisionHasChildWithoutProvision.requiresMultibindingsBoundInParentAndChild() ) .isEqualTo(BOUND_IN_PARENT_AND_CHILD_PROVIDED_BY_PARENT) // Grandchild assertThat( parentWithProvisionHasChildWithoutProvision .childWithoutProvision() .grandchild() .requiresMultibindingsBoundInParent() ) .isEqualTo(BOUND_IN_PARENT) assertThat( parentWithProvisionHasChildWithoutProvision .childWithoutProvision() .grandchild() .requiresMultibindingsBoundInChild() ) .isEqualTo(BOUND_IN_CHILD) assertThat( parentWithProvisionHasChildWithoutProvision .childWithoutProvision() .grandchild() .requiresMultibindingsBoundInParentAndChild() ) .isEqualTo(BOUND_IN_PARENT_AND_CHILD) // Even though the multibinding for Set does not itself have a // contribution from the child, it must be pushed down to (not duplicated in) the child because // its contribution depends on multibindings that have one contribution from the parent and one // from the child (see b/22821341). assertThat( parentWithProvisionHasChildWithoutProvision .childWithoutProvision() .grandchild() .setOfRequiresMultibindingsInParentAndChild() ) .containsExactly(BOUND_IN_PARENT_AND_CHILD) } @Test fun testParentWithProvisionHasChildWithProvision() { // Parent assertThat(parentWithProvisionHasChildWithProvision.requiresMultibindingsBoundInParent()) .isEqualTo(BOUND_IN_PARENT) // Child assertThat( parentWithProvisionHasChildWithProvision .childWithProvision() .requiresMultibindingsBoundInParent() ) .isEqualTo(BOUND_IN_PARENT) assertThat( parentWithProvisionHasChildWithProvision .childWithProvision() .requiresMultibindingsBoundInChild() ) .isEqualTo(BOUND_IN_CHILD) assertThat( parentWithProvisionHasChildWithProvision .childWithProvision() .requiresMultibindingsBoundInParentAndChild() ) .isEqualTo(BOUND_IN_PARENT_AND_CHILD) // https://github.com/google/dagger/issues/401 assertThat( DaggerMultibindingSubcomponents_ParentWithProvisionHasChildWithBinds.create() .childWithBinds() .requiresMultibindingsBoundInParentAndChild() ) .isEqualTo(BOUND_IN_PARENT_AND_CHILD) // Even though the multibinding for Set does not itself have a // contribution from the child, it must be pushed down to (not duplicated in) the child because // its contribution depends on multibindings that have one contribution from the parent and one // from the child (see b/22821341). assertThat( parentWithProvisionHasChildWithProvision .childWithProvision() .setOfRequiresMultibindingsInParentAndChild() ) .containsExactly(BOUND_IN_PARENT_AND_CHILD) // Grandchild assertThat( parentWithProvisionHasChildWithProvision .childWithProvision() .grandchild() .requiresMultibindingsBoundInParent() ) .isEqualTo(BOUND_IN_PARENT) assertThat( parentWithProvisionHasChildWithProvision .childWithProvision() .grandchild() .requiresMultibindingsBoundInChild() ) .isEqualTo(BOUND_IN_CHILD) assertThat( parentWithProvisionHasChildWithProvision .childWithProvision() .grandchild() .requiresMultibindingsBoundInParentAndChild() ) .isEqualTo(BOUND_IN_PARENT_AND_CHILD) // Even though the multibinding for Set does not itself have a // contribution from the child, it must be pushed down to (not duplicated in) the child because // its contribution depends on multibindings that have one contribution from the parent and one // from the child (see b/22821341). assertThat( parentWithProvisionHasChildWithProvision .childWithProvision() .grandchild() .setOfRequiresMultibindingsInParentAndChild() ) .containsExactly(BOUND_IN_PARENT_AND_CHILD) } companion object { private val BOUND_IN_PARENT = RequiresMultibindings( ImmutableSet.of(BoundInParent.INSTANCE), ImmutableMap.of("parent key", BoundInParent.INSTANCE) ) private val BOUND_IN_CHILD = RequiresMultibindings( ImmutableSet.of(BoundInChild.INSTANCE), ImmutableMap.of("child key", BoundInChild.INSTANCE) ) private val BOUND_IN_PARENT_AND_CHILD = RequiresMultibindings( ImmutableSet.of(BoundInParentAndChild.IN_PARENT, BoundInParentAndChild.IN_CHILD), ImmutableMap.of( "parent key", BoundInParentAndChild.IN_PARENT, "child key", BoundInParentAndChild.IN_CHILD ) ) private val BOUND_IN_PARENT_AND_CHILD_PROVIDED_BY_PARENT = RequiresMultibindings( ImmutableSet.of(BoundInParentAndChild.IN_PARENT), ImmutableMap.of("parent key", BoundInParentAndChild.IN_PARENT) ) } }