| /* |
| * Copyright (C) 2020 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.hilt.android.simple; |
| |
| import static androidx.test.core.app.ApplicationProvider.getApplicationContext; |
| import static com.google.common.truth.Truth.assertThat; |
| import static org.junit.Assert.assertThrows; |
| |
| import androidx.test.core.app.ApplicationProvider; |
| import androidx.test.ext.junit.runners.AndroidJUnit4; |
| import dagger.hilt.EntryPoint; |
| import dagger.hilt.EntryPoints; |
| import dagger.hilt.InstallIn; |
| import dagger.hilt.android.testing.BindValue; |
| import dagger.hilt.android.testing.HiltAndroidRule; |
| import dagger.hilt.android.testing.HiltAndroidTest; |
| import dagger.hilt.android.testing.OnComponentReadyRunner; |
| import dagger.hilt.components.SingletonComponent; |
| import java.util.concurrent.atomic.AtomicReference; |
| import org.junit.Rule; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| |
| /** A simple test using Hilt. */ |
| @HiltAndroidTest |
| @RunWith(AndroidJUnit4.class) |
| public final class DelayComponentReadyTest { |
| |
| private static final String EXPECTED_VALUE = "expected"; |
| |
| @EntryPoint |
| @InstallIn(SingletonComponent.class) |
| public interface FooEntryPoint { |
| String foo(); |
| } |
| |
| @Rule public HiltAndroidRule rule = new HiltAndroidRule(this).delayComponentReady(); |
| |
| @BindValue String foo; |
| |
| @Test |
| public void testLateBindValue() throws Exception { |
| AtomicReference<String> fooRef = new AtomicReference<>(); |
| OnComponentReadyRunner.addListener( |
| ApplicationProvider.getApplicationContext(), |
| FooEntryPoint.class, |
| entryPoint -> fooRef.set(entryPoint.foo())); |
| |
| // Test that setting the listener before the component is ready doesn't run the listener. |
| assertThat(fooRef.get()).isNull(); |
| |
| foo = EXPECTED_VALUE; |
| rule.componentReady().inject(); |
| assertThat(EntryPoints.get(getApplicationContext(), FooEntryPoint.class).foo()) |
| .isEqualTo(EXPECTED_VALUE); |
| } |
| |
| @Test |
| public void testUnitializedBindValue_fails() throws Exception { |
| OnComponentReadyRunner.addListener( |
| ApplicationProvider.getApplicationContext(), FooEntryPoint.class, FooEntryPoint::foo); |
| |
| // foo not set |
| NullPointerException expected = assertThrows(NullPointerException.class, rule::componentReady); |
| // This is not the best error message, but it is equivalent to the message from a regular |
| // (non-delayed) @BindValue returning null; |
| assertThat(expected) |
| .hasMessageThat() |
| .isEqualTo("Cannot return null from a non-@Nullable @Provides method"); |
| } |
| |
| @Test |
| public void testDoubleComponentReady_fails() throws Exception { |
| foo = EXPECTED_VALUE; |
| rule.componentReady(); |
| IllegalStateException expected = |
| assertThrows(IllegalStateException.class, rule::componentReady); |
| assertThat(expected).hasMessageThat().isEqualTo("Called componentReady() multiple times"); |
| } |
| |
| @Test |
| public void testMissingComponentReady_fails() throws Exception { |
| // componentReady not called |
| foo = EXPECTED_VALUE; |
| IllegalStateException expected = assertThrows(IllegalStateException.class, rule::inject); |
| assertThat(expected) |
| .hasMessageThat() |
| .isEqualTo("Called inject() before calling componentReady()"); |
| } |
| |
| @Test |
| public void testDelayComponentReadyAfterStart_fails() throws Exception { |
| IllegalStateException expected = |
| assertThrows(IllegalStateException.class, rule::delayComponentReady); |
| assertThat(expected) |
| .hasMessageThat() |
| .isEqualTo("Called delayComponentReady after test execution started"); |
| // Prevents failure due to never calling componentReady() |
| foo = EXPECTED_VALUE; |
| rule.componentReady(); |
| } |
| } |