| /* |
| * Copyright 2019 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.mediaroutertest; |
| |
| import static android.media.MediaRoute2Info.PLAYBACK_VOLUME_FIXED; |
| import static android.media.MediaRoute2Info.PLAYBACK_VOLUME_VARIABLE; |
| |
| import static org.junit.Assert.assertEquals; |
| import static org.junit.Assert.assertNotEquals; |
| import static org.junit.Assert.assertNotNull; |
| import static org.junit.Assert.assertTrue; |
| |
| import android.content.Context; |
| import android.content.Intent; |
| import android.media.MediaRoute2Info; |
| import android.media.MediaRouter2; |
| import android.media.MediaRouter2.RouteCallback; |
| import android.media.MediaRouter2.SessionCallback; |
| import android.media.MediaRouter2Manager; |
| import android.media.RouteDiscoveryRequest; |
| import android.support.test.InstrumentationRegistry; |
| import android.support.test.filters.SmallTest; |
| import android.support.test.runner.AndroidJUnit4; |
| import android.text.TextUtils; |
| |
| import org.junit.After; |
| import org.junit.Before; |
| import org.junit.Ignore; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| |
| import java.util.ArrayList; |
| import java.util.HashMap; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.concurrent.CountDownLatch; |
| import java.util.concurrent.Executor; |
| import java.util.concurrent.Executors; |
| import java.util.concurrent.TimeUnit; |
| import java.util.function.Predicate; |
| |
| @RunWith(AndroidJUnit4.class) |
| @SmallTest |
| public class MediaRouterManagerTest { |
| private static final String TAG = "MediaRouterManagerTest"; |
| |
| public static final String SAMPLE_PROVIDER_ROUTES_ID_PREFIX = |
| "com.android.mediarouteprovider.example/.SampleMediaRoute2ProviderService:"; |
| |
| // Must be the same as SampleMediaRoute2ProviderService except the prefix of IDs. |
| public static final String ROUTE_ID1 = SAMPLE_PROVIDER_ROUTES_ID_PREFIX + "route_id1"; |
| public static final String ROUTE_NAME1 = "Sample Route 1"; |
| public static final String ROUTE_ID2 = SAMPLE_PROVIDER_ROUTES_ID_PREFIX + "route_id2"; |
| public static final String ROUTE_NAME2 = "Sample Route 2"; |
| public static final String ROUTE_ID3_SESSION_CREATION_FAILED = |
| SAMPLE_PROVIDER_ROUTES_ID_PREFIX + "route_id3_session_creation_failed"; |
| public static final String ROUTE_NAME3 = "Sample Route 3 - Session creation failed"; |
| public static final String ROUTE_ID4_TO_SELECT_AND_DESELECT = |
| SAMPLE_PROVIDER_ROUTES_ID_PREFIX + "route_id4_to_select_and_deselect"; |
| public static final String ROUTE_NAME4 = "Sample Route 4 - Route to select and deselect"; |
| public static final String ROUTE_ID5_TO_TRANSFER_TO = |
| SAMPLE_PROVIDER_ROUTES_ID_PREFIX + "route_id5_to_transfer_to"; |
| public static final String ROUTE_NAME5 = "Sample Route 5 - Route to transfer to"; |
| |
| public static final String ROUTE_ID_SPECIAL_TYPE = |
| SAMPLE_PROVIDER_ROUTES_ID_PREFIX + "route_special_type"; |
| public static final String ROUTE_NAME_SPECIAL_TYPE = "Special Type Route"; |
| |
| public static final String SYSTEM_PROVIDER_ID = |
| "com.android.server.media/.SystemMediaRoute2Provider"; |
| |
| public static final int VOLUME_MAX = 100; |
| public static final String ROUTE_ID_FIXED_VOLUME = |
| SAMPLE_PROVIDER_ROUTES_ID_PREFIX + "route_fixed_volume"; |
| public static final String ROUTE_NAME_FIXED_VOLUME = "Fixed Volume Route"; |
| public static final String ROUTE_ID_VARIABLE_VOLUME = |
| SAMPLE_PROVIDER_ROUTES_ID_PREFIX + "route_variable_volume"; |
| public static final String ROUTE_NAME_VARIABLE_VOLUME = "Variable Volume Route"; |
| |
| public static final String ACTION_REMOVE_ROUTE = |
| "com.android.mediarouteprovider.action_remove_route"; |
| |
| public static final String TYPE_SAMPLE = |
| "com.android.mediarouteprovider.TYPE_SAMPLE"; |
| public static final String TYPE_SPECIAL = |
| "com.android.mediarouteprovider.TYPE_SPECIAL"; |
| |
| private static final String TYPE_LIVE_AUDIO = "android.media.intent.route.TYPE_LIVE_AUDIO"; |
| |
| private static final int TIMEOUT_MS = 5000; |
| |
| private Context mContext; |
| private MediaRouter2Manager mManager; |
| private MediaRouter2 mRouter2; |
| private Executor mExecutor; |
| private String mPackageName; |
| |
| private final List<MediaRouter2Manager.Callback> mManagerCallbacks = new ArrayList<>(); |
| private final List<RouteCallback> mRouteCallbacks = new ArrayList<>(); |
| private final List<SessionCallback> mSessionCallbacks = new ArrayList<>(); |
| |
| public static final List<String> TYPES_ALL = new ArrayList(); |
| public static final List<String> TYPES_SPECIAL = new ArrayList(); |
| private static final List<String> TYPES_LIVE_AUDIO = new ArrayList<>(); |
| |
| static { |
| TYPES_ALL.add(TYPE_SAMPLE); |
| TYPES_ALL.add(TYPE_SPECIAL); |
| TYPES_ALL.add(TYPE_LIVE_AUDIO); |
| |
| TYPES_SPECIAL.add(TYPE_SPECIAL); |
| |
| TYPES_LIVE_AUDIO.add(TYPE_LIVE_AUDIO); |
| } |
| |
| @Before |
| public void setUp() throws Exception { |
| mContext = InstrumentationRegistry.getTargetContext(); |
| mManager = MediaRouter2Manager.getInstance(mContext); |
| mRouter2 = MediaRouter2.getInstance(mContext); |
| //TODO: If we need to support thread pool executors, change this to thread pool executor. |
| mExecutor = Executors.newSingleThreadExecutor(); |
| mPackageName = mContext.getPackageName(); |
| } |
| |
| @After |
| public void tearDown() { |
| // unregister callbacks |
| clearCallbacks(); |
| } |
| |
| //TODO: Move to a separate file |
| @Test |
| public void testMediaRoute2Info() { |
| MediaRoute2Info routeInfo1 = new MediaRoute2Info.Builder("id", "name") |
| .build(); |
| MediaRoute2Info routeInfo2 = new MediaRoute2Info.Builder(routeInfo1).build(); |
| |
| MediaRoute2Info routeInfo3 = new MediaRoute2Info.Builder(routeInfo1) |
| .setClientPackageName(mPackageName).build(); |
| |
| assertEquals(routeInfo1, routeInfo2); |
| assertNotEquals(routeInfo1, routeInfo3); |
| } |
| |
| /** |
| * Tests if routes are added correctly when a new callback is registered. |
| */ |
| @Test |
| public void testOnRoutesAdded() throws Exception { |
| CountDownLatch latch = new CountDownLatch(1); |
| addManagerCallback(new MediaRouter2Manager.Callback() { |
| @Override |
| public void onRoutesAdded(List<MediaRoute2Info> routes) { |
| assertTrue(routes.size() > 0); |
| for (MediaRoute2Info route : routes) { |
| if (route.getId().equals(ROUTE_ID1) && route.getName().equals(ROUTE_NAME1)) { |
| latch.countDown(); |
| } |
| } |
| } |
| }); |
| |
| assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS)); |
| } |
| |
| @Test |
| public void testOnRoutesRemoved() throws Exception { |
| CountDownLatch latch = new CountDownLatch(1); |
| Map<String, MediaRoute2Info> routes = waitAndGetRoutesWithManager(TYPES_ALL); |
| |
| addRouterCallback(new RouteCallback()); |
| addManagerCallback(new MediaRouter2Manager.Callback() { |
| @Override |
| public void onRoutesRemoved(List<MediaRoute2Info> routes) { |
| assertTrue(routes.size() > 0); |
| for (MediaRoute2Info route : routes) { |
| if (route.getId().equals(ROUTE_ID2) && route.getName().equals(ROUTE_NAME2)) { |
| latch.countDown(); |
| } |
| } |
| } |
| }); |
| |
| //TODO: Figure out a more proper way to test. |
| // (Control requests shouldn't be used in this way.) |
| mRouter2.sendControlRequest(routes.get(ROUTE_ID2), new Intent(ACTION_REMOVE_ROUTE)); |
| assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS)); |
| } |
| |
| /** |
| * Tests if we get proper routes for application that has special route type. |
| */ |
| @Test |
| public void testRouteType() throws Exception { |
| Map<String, MediaRoute2Info> routes = waitAndGetRoutesWithManager(TYPES_SPECIAL); |
| |
| assertEquals(1, routes.size()); |
| assertNotNull(routes.get(ROUTE_ID_SPECIAL_TYPE)); |
| } |
| |
| /** |
| * Tests if MR2.SessionCallback.onSessionCreated is called |
| * when a route is selected from MR2Manager. |
| */ |
| @Test |
| public void testRouterOnSessionCreated() throws Exception { |
| Map<String, MediaRoute2Info> routes = waitAndGetRoutesWithManager(TYPES_ALL); |
| |
| CountDownLatch latch = new CountDownLatch(1); |
| |
| addManagerCallback(new MediaRouter2Manager.Callback()); |
| //TODO: remove this when it's not necessary. |
| addRouterCallback(new MediaRouter2.RouteCallback()); |
| addSessionCallback(new SessionCallback() { |
| @Override |
| public void onSessionCreated(MediaRouter2.RouteSessionController controller) { |
| if (createRouteMap(controller.getSelectedRoutes()).containsKey(ROUTE_ID1)) { |
| latch.countDown(); |
| } |
| } |
| }); |
| |
| MediaRoute2Info routeToSelect = routes.get(ROUTE_ID1); |
| assertNotNull(routeToSelect); |
| |
| try { |
| mManager.selectRoute(mPackageName, routeToSelect); |
| assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS)); |
| } finally { |
| //TODO: release the session |
| //mManager.selectRoute(mPackageName, null); |
| } |
| } |
| |
| /** |
| * Tests if MR2Manager.Callback.onRouteSelected is called |
| * when a route is selected by MR2Manager. |
| */ |
| @Test |
| @Ignore("TODO: test session created callback instead of onRouteSelected") |
| public void testManagerOnRouteSelected() throws Exception { |
| CountDownLatch latch = new CountDownLatch(1); |
| Map<String, MediaRoute2Info> routes = waitAndGetRoutesWithManager(TYPES_ALL); |
| |
| addRouterCallback(new RouteCallback()); |
| addManagerCallback(new MediaRouter2Manager.Callback() { |
| @Override |
| public void onRouteSelected(String packageName, MediaRoute2Info route) { |
| if (TextUtils.equals(mPackageName, packageName) |
| && route != null && TextUtils.equals(route.getId(), ROUTE_ID1)) { |
| latch.countDown(); |
| } |
| } |
| }); |
| |
| MediaRoute2Info routeToSelect = routes.get(ROUTE_ID1); |
| assertNotNull(routeToSelect); |
| |
| try { |
| mManager.selectRoute(mPackageName, routeToSelect); |
| assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS)); |
| } finally { |
| //TODO: release the session |
| //mManager.selectRoute(mPackageName, null); |
| } |
| } |
| |
| @Test |
| @Ignore("TODO: enable this when 'releasing session' is implemented") |
| public void testGetActiveRoutes() throws Exception { |
| CountDownLatch latch = new CountDownLatch(1); |
| |
| Map<String, MediaRoute2Info> routes = waitAndGetRoutesWithManager(TYPES_ALL); |
| addRouterCallback(new RouteCallback()); |
| addManagerCallback(new MediaRouter2Manager.Callback() { |
| @Override |
| public void onRouteSelected(String packageName, MediaRoute2Info route) { |
| if (TextUtils.equals(mPackageName, packageName) |
| && route != null && TextUtils.equals(route.getId(), ROUTE_ID1)) { |
| latch.countDown(); |
| } |
| } |
| }); |
| |
| //TODO: it fails due to not releasing session |
| assertEquals(0, mManager.getActiveSessions().size()); |
| |
| mManager.selectRoute(mPackageName, routes.get(ROUTE_ID1)); |
| latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS); |
| |
| assertEquals(1, mManager.getActiveSessions().size()); |
| |
| //TODO: release the session |
| /* |
| awaitOnRouteChangedManager( |
| () -> mManager.selectRoute(mPackageName, null), |
| ROUTE_ID1, |
| route -> TextUtils.equals(route.getClientPackageName(), null)); |
| assertEquals(0, mManager.getActiveRoutes().size()); |
| */ |
| } |
| |
| /** |
| * Tests selecting and unselecting routes of a single provider. |
| */ |
| @Test |
| @Ignore("TODO: enable when session is released") |
| public void testSingleProviderSelect() throws Exception { |
| Map<String, MediaRoute2Info> routes = waitAndGetRoutesWithManager(TYPES_ALL); |
| addRouterCallback(new RouteCallback()); |
| |
| awaitOnRouteChangedManager( |
| () -> mManager.selectRoute(mPackageName, routes.get(ROUTE_ID1)), |
| ROUTE_ID1, |
| route -> TextUtils.equals(route.getClientPackageName(), mPackageName)); |
| |
| awaitOnRouteChangedManager( |
| () -> mManager.selectRoute(mPackageName, routes.get(ROUTE_ID2)), |
| ROUTE_ID2, |
| route -> TextUtils.equals(route.getClientPackageName(), mPackageName)); |
| |
| //TODO: release the session |
| /* |
| awaitOnRouteChangedManager( |
| () -> mManager.selectRoute(mPackageName, null), |
| ROUTE_ID2, |
| route -> TextUtils.equals(route.getClientPackageName(), null)); |
| |
| */ |
| } |
| |
| @Test |
| public void testControlVolumeWithManager() throws Exception { |
| Map<String, MediaRoute2Info> routes = waitAndGetRoutesWithManager(TYPES_ALL); |
| MediaRoute2Info volRoute = routes.get(ROUTE_ID_VARIABLE_VOLUME); |
| |
| int originalVolume = volRoute.getVolume(); |
| int deltaVolume = (originalVolume == volRoute.getVolumeMax() ? -1 : 1); |
| |
| awaitOnRouteChangedManager( |
| () -> mManager.requestUpdateVolume(volRoute, deltaVolume), |
| ROUTE_ID_VARIABLE_VOLUME, |
| (route -> route.getVolume() == originalVolume + deltaVolume)); |
| |
| awaitOnRouteChangedManager( |
| () -> mManager.requestSetVolume(volRoute, originalVolume), |
| ROUTE_ID_VARIABLE_VOLUME, |
| (route -> route.getVolume() == originalVolume)); |
| } |
| |
| @Test |
| public void testVolumeHandling() throws Exception { |
| Map<String, MediaRoute2Info> routes = waitAndGetRoutesWithManager(TYPES_ALL); |
| |
| MediaRoute2Info fixedVolumeRoute = routes.get(ROUTE_ID_FIXED_VOLUME); |
| MediaRoute2Info variableVolumeRoute = routes.get(ROUTE_ID_VARIABLE_VOLUME); |
| |
| assertEquals(PLAYBACK_VOLUME_FIXED, fixedVolumeRoute.getVolumeHandling()); |
| assertEquals(PLAYBACK_VOLUME_VARIABLE, variableVolumeRoute.getVolumeHandling()); |
| assertEquals(VOLUME_MAX, variableVolumeRoute.getVolumeMax()); |
| } |
| |
| Map<String, MediaRoute2Info> waitAndGetRoutesWithManager(List<String> routeTypes) |
| throws Exception { |
| CountDownLatch latch = new CountDownLatch(2); |
| |
| // A dummy callback is required to send route type info. |
| RouteCallback routeCallback = new RouteCallback(); |
| MediaRouter2Manager.Callback managerCallback = new MediaRouter2Manager.Callback() { |
| @Override |
| public void onRoutesAdded(List<MediaRoute2Info> routes) { |
| for (int i = 0; i < routes.size(); i++) { |
| //TODO: use isSystem() or similar method when it's ready |
| if (!TextUtils.equals(routes.get(i).getProviderId(), SYSTEM_PROVIDER_ID)) { |
| latch.countDown(); |
| break; |
| } |
| } |
| } |
| |
| @Override |
| public void onControlCategoriesChanged(String packageName, List<String> routeTypes) { |
| if (TextUtils.equals(mPackageName, packageName) |
| && routeTypes.equals(routeTypes)) { |
| latch.countDown(); |
| } |
| } |
| }; |
| mManager.registerCallback(mExecutor, managerCallback); |
| mRouter2.registerRouteCallback(mExecutor, routeCallback, |
| new RouteDiscoveryRequest.Builder(routeTypes, true).build()); |
| try { |
| latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS); |
| return createRouteMap(mManager.getAvailableRoutes(mPackageName)); |
| } finally { |
| mRouter2.unregisterRouteCallback(routeCallback); |
| mManager.unregisterCallback(managerCallback); |
| } |
| } |
| |
| void awaitOnRouteChangedManager(Runnable task, String routeId, |
| Predicate<MediaRoute2Info> predicate) throws Exception { |
| CountDownLatch latch = new CountDownLatch(1); |
| MediaRouter2Manager.Callback callback = new MediaRouter2Manager.Callback() { |
| @Override |
| public void onRoutesChanged(List<MediaRoute2Info> changed) { |
| MediaRoute2Info route = createRouteMap(changed).get(routeId); |
| if (route != null && predicate.test(route)) { |
| latch.countDown(); |
| } |
| } |
| }; |
| mManager.registerCallback(mExecutor, callback); |
| try { |
| task.run(); |
| assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS)); |
| } finally { |
| mManager.unregisterCallback(callback); |
| } |
| } |
| |
| // Helper for getting routes easily |
| static Map<String, MediaRoute2Info> createRouteMap(List<MediaRoute2Info> routes) { |
| Map<String, MediaRoute2Info> routeMap = new HashMap<>(); |
| for (MediaRoute2Info route : routes) { |
| routeMap.put(route.getId(), route); |
| } |
| return routeMap; |
| } |
| |
| private void addManagerCallback(MediaRouter2Manager.Callback callback) { |
| mManagerCallbacks.add(callback); |
| mManager.registerCallback(mExecutor, callback); |
| } |
| |
| private void addRouterCallback(RouteCallback routeCallback) { |
| mRouteCallbacks.add(routeCallback); |
| mRouter2.registerRouteCallback(mExecutor, routeCallback, RouteDiscoveryRequest.EMPTY); |
| } |
| |
| private void addSessionCallback(SessionCallback sessionCallback) { |
| mSessionCallbacks.add(sessionCallback); |
| mRouter2.registerSessionCallback(mExecutor, sessionCallback); |
| } |
| |
| private void clearCallbacks() { |
| for (MediaRouter2Manager.Callback callback : mManagerCallbacks) { |
| mManager.unregisterCallback(callback); |
| } |
| mManagerCallbacks.clear(); |
| |
| for (RouteCallback routeCallback : mRouteCallbacks) { |
| mRouter2.unregisterRouteCallback(routeCallback); |
| } |
| mRouteCallbacks.clear(); |
| |
| for (SessionCallback sessionCallback : mSessionCallbacks) { |
| mRouter2.unregisterSessionCallback(sessionCallback); |
| } |
| mSessionCallbacks.clear(); |
| } |
| } |