| /* |
| * Copyright (C) 2017 The Android Open Source Project |
| * |
| * 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 com.android.systemui.statusbar; |
| |
| import static org.junit.Assert.assertArrayEquals; |
| import static org.mockito.ArgumentMatchers.any; |
| import static org.mockito.Mockito.doAnswer; |
| import static org.mockito.Mockito.never; |
| import static org.mockito.Mockito.times; |
| import static org.mockito.Mockito.verify; |
| import static org.mockito.Mockito.when; |
| |
| import android.app.Notification; |
| import android.os.Handler; |
| import android.os.Looper; |
| import android.os.RemoteException; |
| import android.os.UserHandle; |
| import android.service.notification.StatusBarNotification; |
| import android.testing.AndroidTestingRunner; |
| import android.testing.TestableLooper; |
| |
| import androidx.test.filters.SmallTest; |
| |
| import com.android.internal.statusbar.IStatusBarService; |
| import com.android.internal.statusbar.NotificationVisibility; |
| import com.android.systemui.SysuiTestCase; |
| |
| import com.google.android.collect.Lists; |
| |
| import org.junit.Before; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| import org.mockito.Mock; |
| import org.mockito.Mockito; |
| import org.mockito.MockitoAnnotations; |
| import org.mockito.stubbing.Answer; |
| |
| import java.util.concurrent.ConcurrentLinkedQueue; |
| |
| @SmallTest |
| @RunWith(AndroidTestingRunner.class) |
| @TestableLooper.RunWithLooper(setAsMainLooper = true) |
| public class NotificationLoggerTest extends SysuiTestCase { |
| private static final String TEST_PACKAGE_NAME = "test"; |
| private static final int TEST_UID = 0; |
| |
| @Mock private NotificationPresenter mPresenter; |
| @Mock private NotificationListContainer mListContainer; |
| @Mock private IStatusBarService mBarService; |
| @Mock private NotificationData mNotificationData; |
| @Mock private ExpandableNotificationRow mRow; |
| |
| // Dependency mocks: |
| @Mock private NotificationEntryManager mEntryManager; |
| @Mock private NotificationListener mListener; |
| |
| private NotificationData.Entry mEntry; |
| private StatusBarNotification mSbn; |
| private TestableNotificationLogger mLogger; |
| private ConcurrentLinkedQueue<AssertionError> mErrorQueue = new ConcurrentLinkedQueue<>(); |
| |
| @Before |
| public void setUp() throws RemoteException { |
| MockitoAnnotations.initMocks(this); |
| mDependency.injectTestDependency(NotificationEntryManager.class, mEntryManager); |
| mDependency.injectTestDependency(NotificationListener.class, mListener); |
| |
| when(mEntryManager.getNotificationData()).thenReturn(mNotificationData); |
| |
| mSbn = new StatusBarNotification(TEST_PACKAGE_NAME, TEST_PACKAGE_NAME, 0, null, TEST_UID, |
| 0, new Notification(), UserHandle.CURRENT, null, 0); |
| mEntry = new NotificationData.Entry(mSbn); |
| mEntry.row = mRow; |
| |
| mLogger = new TestableNotificationLogger(mBarService); |
| mLogger.setUpWithEntryManager(mEntryManager, mListContainer); |
| } |
| |
| @Test |
| public void testOnChildLocationsChangedReportsVisibilityChanged() throws Exception { |
| NotificationVisibility[] newlyVisibleKeys = { |
| NotificationVisibility.obtain(mEntry.key, 0, 1, true) |
| }; |
| NotificationVisibility[] noLongerVisibleKeys = {}; |
| doAnswer((Answer) invocation -> { |
| try { |
| assertArrayEquals(newlyVisibleKeys, |
| (NotificationVisibility[]) invocation.getArguments()[0]); |
| assertArrayEquals(noLongerVisibleKeys, |
| (NotificationVisibility[]) invocation.getArguments()[1]); |
| } catch (AssertionError error) { |
| mErrorQueue.offer(error); |
| } |
| return null; |
| } |
| ).when(mBarService).onNotificationVisibilityChanged(any(NotificationVisibility[].class), |
| any(NotificationVisibility[].class)); |
| |
| when(mListContainer.isInVisibleLocation(any())).thenReturn(true); |
| when(mNotificationData.getActiveNotifications()).thenReturn(Lists.newArrayList(mEntry)); |
| mLogger.getChildLocationsChangedListenerForTest().onChildLocationsChanged(); |
| TestableLooper.get(this).processAllMessages(); |
| waitForUiOffloadThread(); |
| |
| if(!mErrorQueue.isEmpty()) { |
| throw mErrorQueue.poll(); |
| } |
| |
| // |mEntry| won't change visibility, so it shouldn't be reported again: |
| Mockito.reset(mBarService); |
| mLogger.getChildLocationsChangedListenerForTest().onChildLocationsChanged(); |
| TestableLooper.get(this).processAllMessages(); |
| waitForUiOffloadThread(); |
| |
| verify(mBarService, never()).onNotificationVisibilityChanged(any(), any()); |
| } |
| |
| @Test |
| public void testStoppingNotificationLoggingReportsCurrentNotifications() |
| throws Exception { |
| when(mListContainer.isInVisibleLocation(any())).thenReturn(true); |
| when(mNotificationData.getActiveNotifications()).thenReturn(Lists.newArrayList(mEntry)); |
| mLogger.getChildLocationsChangedListenerForTest().onChildLocationsChanged(); |
| TestableLooper.get(this).processAllMessages(); |
| waitForUiOffloadThread(); |
| Mockito.reset(mBarService); |
| |
| mLogger.stopNotificationLogging(); |
| waitForUiOffloadThread(); |
| // The visibility objects are recycled by NotificationLogger, so we can't use specific |
| // matchers here. |
| verify(mBarService, times(1)).onNotificationVisibilityChanged(any(), any()); |
| } |
| |
| private class TestableNotificationLogger extends NotificationLogger { |
| |
| public TestableNotificationLogger(IStatusBarService barService) { |
| mBarService = barService; |
| // Make this on the current thread so we can wait for it during tests. |
| mHandler = Handler.createAsync(Looper.myLooper()); |
| } |
| |
| public OnChildLocationsChangedListener |
| getChildLocationsChangedListenerForTest() { |
| return mNotificationLocationsChangedListener; |
| } |
| |
| public Handler getHandlerForTest() { |
| return mHandler; |
| } |
| } |
| } |