MediaController2Test.java revision ebc904f04b5f4c011ec791b1b0e94660ac02fdba
1fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang/*
2fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * Copyright 2018 The Android Open Source Project
3fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang *
4fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * Licensed under the Apache License, Version 2.0 (the "License");
5fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * you may not use this file except in compliance with the License.
6fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * You may obtain a copy of the License at
7fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang *
8fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang *      http://www.apache.org/licenses/LICENSE-2.0
9fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang *
10fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * Unless required by applicable law or agreed to in writing, software
11fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * distributed under the License is distributed on an "AS IS" BASIS,
12fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * See the License for the specific language governing permissions and
14fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * limitations under the License.
15fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang */
16fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
17fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangpackage androidx.media;
18fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
19e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moonimport static org.junit.Assert.assertEquals;
20fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport static org.junit.Assert.assertFalse;
21fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport static org.junit.Assert.assertNotEquals;
22fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport static org.junit.Assert.assertNotNull;
23fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport static org.junit.Assert.assertNull;
24fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport static org.junit.Assert.assertTrue;
25fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport static org.junit.Assert.fail;
26fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
27fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.app.PendingIntent;
28fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.content.Intent;
29fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.media.AudioManager;
30fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.net.Uri;
31fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.os.Build;
32fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.os.Bundle;
33fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.os.Handler;
34fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.os.HandlerThread;
35fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.os.Process;
36fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.os.ResultReceiver;
37fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.support.test.filters.FlakyTest;
38fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.support.test.filters.SdkSuppress;
39fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.support.test.filters.SmallTest;
40fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport android.support.test.runner.AndroidJUnit4;
41fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
42fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport androidx.annotation.NonNull;
43fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport androidx.media.MediaController2.ControllerCallback;
44e57a84cc8a3c62a777d4dcf48d1c7f6e58657bf6Jaewan Kimimport androidx.media.MediaLibraryService2.MediaLibrarySession.MediaLibrarySessionCallback;
45fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport androidx.media.MediaSession2.ControllerInfo;
46fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport androidx.media.MediaSession2.SessionCallback;
47fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport androidx.media.TestServiceRegistry.SessionServiceCallback;
48fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport androidx.media.TestUtils.SyncHandler;
49fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
50fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport org.junit.After;
51fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport org.junit.Before;
52fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport org.junit.Ignore;
53fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport org.junit.Test;
54fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport org.junit.runner.RunWith;
55fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
56fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.lang.reflect.Method;
57fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.util.List;
58fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.util.concurrent.CountDownLatch;
59fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.util.concurrent.TimeUnit;
60fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.util.concurrent.atomic.AtomicReference;
61fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
62fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang/**
63fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * Tests {@link MediaController2}.
64fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang */
65fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang// TODO(jaewan): Implement host-side test so controller and session can run in different processes.
66fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang// TODO(jaewan): Fix flaky failure -- see MediaController2Impl.getController()
67fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang// TODO(jaeawn): Revisit create/close session in the sHandler. It's no longer necessary.
68fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang@SdkSuppress(minSdkVersion = Build.VERSION_CODES.KITKAT)
69fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang@RunWith(AndroidJUnit4.class)
70fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang@SmallTest
71fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang@FlakyTest
72fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangpublic class MediaController2Test extends MediaSession2TestBase {
73fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    private static final String TAG = "MediaController2Test";
74fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
75fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    PendingIntent mIntent;
76fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    MediaSession2 mSession;
77fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    MediaController2 mController;
78fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    MockPlayer mPlayer;
79fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    MockPlaylistAgent mMockAgent;
80fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
81fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Before
82fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Override
83fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void setUp() throws Exception {
84fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        super.setUp();
85fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Intent sessionActivity = new Intent(mContext, MockActivity.class);
86fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Create this test specific MediaSession2 to use our own Handler.
87fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mIntent = PendingIntent.getActivity(mContext, 0, sessionActivity, 0);
88fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
89fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mPlayer = new MockPlayer(1);
90fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mMockAgent = new MockPlaylistAgent();
91fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession = new MediaSession2.Builder(mContext)
92fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
93fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlaylistAgent(mMockAgent)
94fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, new SessionCallback() {
95fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    @Override
96fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    public SessionCommandGroup2 onConnect(MediaSession2 session,
97fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            ControllerInfo controller) {
98fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        if (Process.myUid() == controller.getUid()) {
99fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            return super.onConnect(session, controller);
100fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        }
101fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        return null;
102fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
103fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
104fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    @Override
105fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    public void onPlaylistMetadataChanged(MediaSession2 session,
106fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            MediaPlaylistAgent playlistAgent,
107fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            MediaMetadata2 metadata) {
108fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        super.onPlaylistMetadataChanged(session, playlistAgent, metadata);
109fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
110fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                })
111fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionActivity(mIntent)
112fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId(TAG).build();
113fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController = createController(mSession.getToken());
114fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestServiceRegistry.getInstance().setHandler(sHandler);
115fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
116fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
117fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @After
118fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Override
119fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void cleanUp() throws Exception {
120fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        super.cleanUp();
121fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        if (mSession != null) {
122fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mSession.close();
123fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
124fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestServiceRegistry.getInstance().cleanUp();
125fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
126fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
127fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
128fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * Test if the {@link MediaSession2TestBase.TestControllerCallback} wraps the callback proxy
129fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * without missing any method.
130fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
131fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
132fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testTestControllerCallback() {
133fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
134fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        Method[] methods = TestControllerCallback.class.getMethods();
135fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(methods);
136fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        for (int i = 0; i < methods.length; i++) {
137fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            // For any methods in the controller callback, TestControllerCallback should have
138fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            // overriden the method and call matching API in the callback proxy.
139fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertNotEquals("TestControllerCallback should override " + methods[i]
140fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            + " and call callback proxy",
141fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    ControllerCallback.class, methods[i].getDeclaringClass());
142fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
143fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
144fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
145fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
146fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPlay() {
147fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
148fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.play();
149fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
150fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
151fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
152fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
153fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
154fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mPlayCalled);
155fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
156fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
157fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
158fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPause() {
159fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
160fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.pause();
161fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
162fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
163fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
164fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
165fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
166fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mPauseCalled);
167fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
168fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
169fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
170fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testReset() {
171fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
172fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.reset();
173fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
174fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
175fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
176fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
177fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
178fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mResetCalled);
179fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
180fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
181fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
182fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPrepare() {
183fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
184fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.prepare();
185fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
186fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
187fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
188fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
189fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
190fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mPrepareCalled);
191fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
192fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
193fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
194fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSeekTo() {
195fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
196fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final long seekPosition = 12125L;
197fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.seekTo(seekPosition);
198fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
199fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
200fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
201fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
202fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
203fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mSeekToCalled);
204fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(seekPosition, mPlayer.mSeekPosition);
205fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
206fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
207fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
208fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGettersAfterConnected() throws InterruptedException {
209fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
210fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int state = MediaPlayerBase.PLAYER_STATE_PLAYING;
211ebc904f04b5f4c011ec791b1b0e94660ac02fdbaHyundo Moon        final int bufferingState = MediaPlayerBase.BUFFERING_STATE_BUFFERING_COMPLETE;
212fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final long position = 150000;
213fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final long bufferedPosition = 900000;
214fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final float speed = 0.5f;
2153a151f1283d08b40c12e4a17903ba421f23c4342Hyundo Moon        final MediaItem2 currentMediaItem = TestUtils.createMediaItemWithMetadata();
216fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
217fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mPlayer.mLastPlayerState = state;
218ebc904f04b5f4c011ec791b1b0e94660ac02fdbaHyundo Moon        mPlayer.mLastBufferingState = bufferingState;
219fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mPlayer.mCurrentPosition = position;
220fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mPlayer.mBufferedPosition = bufferedPosition;
221fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mPlayer.mPlaybackSpeed = speed;
2223a151f1283d08b40c12e4a17903ba421f23c4342Hyundo Moon        mMockAgent.mCurrentMediaItem = currentMediaItem;
223fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
2241e7820def08357c7a191557bccc34e435c9cf6d4Jaewan Kim        long time1 = System.currentTimeMillis();
225fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken());
2261e7820def08357c7a191557bccc34e435c9cf6d4Jaewan Kim        long time2 = System.currentTimeMillis();
227fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(state, controller.getPlayerState());
228fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(bufferedPosition, controller.getBufferedPosition());
229e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        assertEquals(speed, controller.getPlaybackSpeed(), 0.0f);
2301e7820def08357c7a191557bccc34e435c9cf6d4Jaewan Kim        long positionLowerBound = (long) (position + speed * (System.currentTimeMillis() - time2));
2311e7820def08357c7a191557bccc34e435c9cf6d4Jaewan Kim        long currentPosition = controller.getCurrentPosition();
2321e7820def08357c7a191557bccc34e435c9cf6d4Jaewan Kim        long positionUpperBound = (long) (position + speed * (System.currentTimeMillis() - time1));
2331e7820def08357c7a191557bccc34e435c9cf6d4Jaewan Kim        assertTrue("curPos=" + currentPosition + ", lowerBound=" + positionLowerBound
2341e7820def08357c7a191557bccc34e435c9cf6d4Jaewan Kim                        + ", upperBound=" + positionUpperBound,
2351e7820def08357c7a191557bccc34e435c9cf6d4Jaewan Kim                positionLowerBound <= currentPosition && currentPosition <= positionUpperBound);
2363a151f1283d08b40c12e4a17903ba421f23c4342Hyundo Moon        assertEquals(currentMediaItem, controller.getCurrentMediaItem());
237fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
238fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
239fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
240fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetSessionActivity() {
241fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
242fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        PendingIntent sessionActivity = mController.getSessionActivity();
243fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(mContext.getPackageName(), sessionActivity.getCreatorPackage());
244fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(Process.myUid(), sessionActivity.getCreatorUid());
245fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
246fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
247fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
248fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetPlaylist() throws InterruptedException {
249fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
250fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final List<MediaItem2> list = TestUtils.createPlaylist(2);
251fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.setPlaylist(list, null /* Metadata */);
252fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
253fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
254fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSetPlaylistCalled);
255fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNull(mMockAgent.mMetadata);
256fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
257fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(mMockAgent.mPlaylist);
258fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(list.size(), mMockAgent.mPlaylist.size());
259fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        for (int i = 0; i < list.size(); i++) {
260fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            // MediaController2.setPlaylist does not ensure the equality of the items.
261fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertEquals(list.get(i).getMediaId(), mMockAgent.mPlaylist.get(i).getMediaId());
262fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
263fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
264fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
265fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
266fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * This also tests {@link ControllerCallback#onPlaylistChanged(
267fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * MediaController2, List, MediaMetadata2)}.
268fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
269fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
270fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetPlaylist() throws InterruptedException {
271fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
272fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final List<MediaItem2> testList = TestUtils.createPlaylist(2);
273fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final AtomicReference<List<MediaItem2>> listFromCallback = new AtomicReference<>();
274fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
275fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
276fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
277fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlaylistChanged(MediaController2 controller,
278fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    List<MediaItem2> playlist, MediaMetadata2 metadata) {
279fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertNotNull(playlist);
280fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testList.size(), playlist.size());
281fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                for (int i = 0; i < playlist.size(); i++) {
282fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    assertEquals(testList.get(i).getMediaId(), playlist.get(i).getMediaId());
283fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
284fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                listFromCallback.set(playlist);
285fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
286fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
287fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
288fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
289fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
290fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public List<MediaItem2> getPlaylist() {
291fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return testList;
292fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
293fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
294fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
295fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
296fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testControllerCallback_onPlaylistChanged")
297fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, new SessionCallback() {})
298fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlaylistAgent(agent)
299fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .build()) {
300fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(
301fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    session.getToken(), true, callback);
302fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            agent.notifyPlaylistChanged();
303fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
304fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertEquals(listFromCallback.get(), controller.getPlaylist());
305fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
306fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
307fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
308fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
309fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testUpdatePlaylistMetadata() throws InterruptedException {
310fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
311fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaMetadata2 testMetadata = TestUtils.createMetadata();
312fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.updatePlaylistMetadata(testMetadata);
313fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
314fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
315fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mUpdatePlaylistMetadataCalled);
316fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(mMockAgent.mMetadata);
317fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testMetadata.getMediaId(), mMockAgent.mMetadata.getMediaId());
318fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
319fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
320fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
321fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetPlaylistMetadata() throws InterruptedException {
322fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
323fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaMetadata2 testMetadata = TestUtils.createMetadata();
324fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final AtomicReference<MediaMetadata2> metadataFromCallback = new AtomicReference<>();
325fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
326fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
327fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
328fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlaylistMetadataChanged(MediaController2 controller,
329fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    MediaMetadata2 metadata) {
330fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertNotNull(testMetadata);
331fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testMetadata.getMediaId(), metadata.getMediaId());
332fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                metadataFromCallback.set(metadata);
333fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
334fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
335fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
336fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
337fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
338fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public MediaMetadata2 getPlaylistMetadata() {
339fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return testMetadata;
340fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
341fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
342fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
343fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
344fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testGetPlaylistMetadata")
345fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, new SessionCallback() {})
346fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlaylistAgent(agent)
347fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .build()) {
348fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken(), true, callback);
349fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            agent.notifyPlaylistMetadataChanged();
350fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
351fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertEquals(metadataFromCallback.get().getMediaId(),
352fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    controller.getPlaylistMetadata().getMediaId());
353fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
354fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
355fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
356e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon    @Test
357e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon    public void testSetPlaybackSpeed() throws Exception {
358e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        prepareLooper();
359e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        final float speed = 1.5f;
360e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        mController.setPlaybackSpeed(speed);
361e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        assertTrue(mPlayer.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
362e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        assertEquals(speed, mPlayer.mPlaybackSpeed, 0.0f);
363e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon    }
364e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon
365fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
366fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * Test whether {@link MediaSession2#setPlaylist(List, MediaMetadata2)} is notified
367fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * through the
368fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * {@link ControllerCallback#onPlaylistMetadataChanged(MediaController2, MediaMetadata2)}
369fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * if the controller doesn't have {@link SessionCommand2#COMMAND_CODE_PLAYLIST_GET_LIST} but
370fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * {@link SessionCommand2#COMMAND_CODE_PLAYLIST_GET_LIST_METADATA}.
371fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
372fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
373fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_onPlaylistMetadataChanged() throws InterruptedException {
374fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
375fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaItem2 item = TestUtils.createMediaItemWithMetadata();
376fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final List<MediaItem2> list = TestUtils.createPlaylist(2);
377fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
378fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
379fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
380fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlaylistMetadataChanged(MediaController2 controller,
381fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    MediaMetadata2 metadata) {
382fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertNotNull(metadata);
383fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(item.getMediaId(), metadata.getMediaId());
384fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
385fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
386fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
387fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback sessionCallback = new SessionCallback() {
388fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
389fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public SessionCommandGroup2 onConnect(MediaSession2 session,
390fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    ControllerInfo controller) {
391fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                if (Process.myUid() == controller.getUid()) {
392fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    SessionCommandGroup2 commands = new SessionCommandGroup2();
393fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    commands.addCommand(new SessionCommand2(
394fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                              SessionCommand2.COMMAND_CODE_PLAYLIST_GET_LIST_METADATA));
395fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    return commands;
396fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
397fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return super.onConnect(session, controller);
398fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
399fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
400fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
401fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
402fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public MediaMetadata2 getPlaylistMetadata() {
403fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return item.getMetadata();
404fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
405fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
406fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
407fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public List<MediaItem2> getPlaylist() {
408fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return list;
409fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
410fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
411fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
412fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
413fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testControllerCallback_onPlaylistMetadataChanged")
414fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, sessionCallback)
415fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlaylistAgent(agent)
416fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .build()) {
417fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken(), true, callback);
418fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            agent.notifyPlaylistMetadataChanged();
419fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
420fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
421fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
422fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
423fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
424fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testAddPlaylistItem() throws InterruptedException {
425fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
426fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testIndex = 12;
427fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaItem2 testMediaItem = TestUtils.createMediaItemWithMetadata();
428fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.addPlaylistItem(testIndex, testMediaItem);
429fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
430fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
431fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mAddPlaylistItemCalled);
432fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testIndex, mMockAgent.mIndex);
433fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // MediaController2.addPlaylistItem does not ensure the equality of the items.
434fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testMediaItem.getMediaId(), mMockAgent.mItem.getMediaId());
435fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
436fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
437fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
438fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testRemovePlaylistItem() throws InterruptedException {
439fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
440fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mMockAgent.mPlaylist = TestUtils.createPlaylist(2);
441fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
442fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Recreate controller for sending removePlaylistItem.
443fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // It's easier to ensure that MediaController2.getPlaylist() returns the playlist from the
444fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // agent.
445fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken());
446fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaItem2 targetItem = controller.getPlaylist().get(0);
447fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.removePlaylistItem(targetItem);
448fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
449fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
450fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mRemovePlaylistItemCalled);
451fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(targetItem, mMockAgent.mItem);
452fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
453fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
454fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
455fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testReplacePlaylistItem() throws InterruptedException {
456fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
457fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testIndex = 12;
458fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaItem2 testMediaItem = TestUtils.createMediaItemWithMetadata();
459fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.replacePlaylistItem(testIndex, testMediaItem);
460fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
461fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
462fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mReplacePlaylistItemCalled);
463fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // MediaController2.replacePlaylistItem does not ensure the equality of the items.
464fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testMediaItem.getMediaId(), mMockAgent.mItem.getMediaId());
465fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
466fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
467fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
468fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSkipToPreviousItem() throws InterruptedException {
469fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
470fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.skipToPreviousItem();
471fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
472fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSkipToPreviousItemCalled);
473fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
474fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
475fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
476fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSkipToNextItem() throws InterruptedException {
477fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
478fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.skipToNextItem();
479fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
480fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSkipToNextItemCalled);
481fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
482fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
483fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
484fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSkipToPlaylistItem() throws InterruptedException {
485fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
486fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken());
487fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaItem2 targetItem = TestUtils.createMediaItemWithMetadata();
488fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.skipToPlaylistItem(targetItem);
489fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
490fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
491fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSkipToPlaylistItemCalled);
492fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(targetItem, mMockAgent.mItem);
493fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
494fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
495fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
496fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * This also tests {@link ControllerCallback#onShuffleModeChanged(MediaController2, int)}.
497fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
498fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
499fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetShuffleMode() throws InterruptedException {
500fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
501fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testShuffleMode = MediaPlaylistAgent.SHUFFLE_MODE_GROUP;
502fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
503fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
504fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public int getShuffleMode() {
505fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return testShuffleMode;
506fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
507fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
508fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
509fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
510fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
511fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onShuffleModeChanged(MediaController2 controller, int shuffleMode) {
512fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testShuffleMode, shuffleMode);
513fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
514fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
515fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
516fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.updatePlayer(mPlayer, agent, null);
517fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken(), true, callback);
518fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        agent.notifyShuffleModeChanged();
519fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
520fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testShuffleMode, controller.getShuffleMode());
521fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
522fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
523fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
524fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetShuffleMode() throws InterruptedException {
525fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
526fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testShuffleMode = MediaPlaylistAgent.SHUFFLE_MODE_GROUP;
527fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.setShuffleMode(testShuffleMode);
528fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
529fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
530fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSetShuffleModeCalled);
531fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testShuffleMode, mMockAgent.mShuffleMode);
532fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
533fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
534fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
535fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * This also tests {@link ControllerCallback#onRepeatModeChanged(MediaController2, int)}.
536fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
537fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
538fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetRepeatMode() throws InterruptedException {
539fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
540fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testRepeatMode = MediaPlaylistAgent.REPEAT_MODE_GROUP;
541fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
542fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
543fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public int getRepeatMode() {
544fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return testRepeatMode;
545fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
546fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
547fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
548fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
549fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
550fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onRepeatModeChanged(MediaController2 controller, int repeatMode) {
551fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testRepeatMode, repeatMode);
552fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
553fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
554fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
555fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.updatePlayer(mPlayer, agent, null);
556fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken(), true, callback);
557fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        agent.notifyRepeatModeChanged();
558fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
559fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testRepeatMode, controller.getRepeatMode());
560fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
561fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
562fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
563fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetRepeatMode() throws InterruptedException {
564fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
565fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testRepeatMode = MediaPlaylistAgent.REPEAT_MODE_GROUP;
566fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.setRepeatMode(testRepeatMode);
567fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
568fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
569fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSetRepeatModeCalled);
570fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testRepeatMode, mMockAgent.mRepeatMode);
571fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
572fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
573fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
574fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetVolumeTo() throws Exception {
575fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO(jaewan): Also test with local volume.
576fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
577fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int maxVolume = 100;
578fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int currentVolume = 23;
579fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int volumeControlType = VolumeProviderCompat.VOLUME_CONTROL_ABSOLUTE;
580fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestVolumeProvider volumeProvider =
581fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                new TestVolumeProvider(volumeControlType, maxVolume, currentVolume);
582fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
583fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.updatePlayer(new MockPlayer(0), null, volumeProvider);
584fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaController2 controller = createController(mSession.getToken(), true, null);
585fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
586fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int targetVolume = 50;
587fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.setVolumeTo(targetVolume, 0 /* flags */);
588fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(volumeProvider.mLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
589fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(volumeProvider.mSetVolumeToCalled);
590fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(targetVolume, volumeProvider.mVolume);
591fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
592fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
593fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
594fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testAdjustVolume() throws Exception {
595fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO(jaewan): Also test with local volume.
596fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
597fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int maxVolume = 100;
598fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int currentVolume = 23;
599fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int volumeControlType = VolumeProviderCompat.VOLUME_CONTROL_ABSOLUTE;
600fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestVolumeProvider volumeProvider =
601fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                new TestVolumeProvider(volumeControlType, maxVolume, currentVolume);
602fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
603fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.updatePlayer(new MockPlayer(0), null, volumeProvider);
604fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaController2 controller = createController(mSession.getToken(), true, null);
605fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
606fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int direction = AudioManager.ADJUST_RAISE;
607fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.adjustVolume(direction, 0 /* flags */);
608fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(volumeProvider.mLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
609fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(volumeProvider.mAdjustVolumeCalled);
610fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(direction, volumeProvider.mDirection);
611fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
612fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
613fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
614fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetPackageName() {
615fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
616fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(mContext.getPackageName(), mController.getSessionToken().getPackageName());
617fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
618fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
619fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
620fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSendCustomCommand() throws InterruptedException {
621fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
622fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO(jaewan): Need to revisit with the permission.
623fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCommand2 testCommand =
624fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                new SessionCommand2(SessionCommand2.COMMAND_CODE_PLAYBACK_PREPARE);
625fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle testArgs = new Bundle();
626fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testArgs.putString("args", "testSendCustomCommand");
627fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
628fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
629fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
630fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
631fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onCustomCommand(MediaSession2 session, ControllerInfo controller,
632fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    SessionCommand2 customCommand, Bundle args, ResultReceiver cb) {
633fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
634fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testCommand, customCommand);
635fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(testArgs, args));
636fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertNull(cb);
637fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
638fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
639fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
640fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.close();
641fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession = new MediaSession2.Builder(mContext).setPlayer(mPlayer)
642fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback).setId(TAG).build();
643fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaController2 controller = createController(mSession.getToken());
644fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.sendCustomCommand(testCommand, testArgs, null);
645fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
646fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
647fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
648fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
649fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_onConnected() throws InterruptedException {
650fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
651fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // createController() uses controller callback to wait until the controller becomes
652fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // available.
653fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken());
654fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(controller);
655fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
656fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
657fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
658fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_sessionRejects() throws InterruptedException {
659fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
660fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaSession2.SessionCallback sessionCallback = new SessionCallback() {
661fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
662fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public SessionCommandGroup2 onConnect(MediaSession2 session,
663fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    ControllerInfo controller) {
664fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return null;
665fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
666fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
667fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        sHandler.postAndSync(new Runnable() {
668fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
669fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void run() {
670fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession.close();
671fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession = new MediaSession2.Builder(mContext).setPlayer(mPlayer)
672fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        .setSessionCallback(sHandlerExecutor, sessionCallback).build();
673fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
674fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
675fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller =
676fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                createController(mSession.getToken(), false, null);
677fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(controller);
678fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForConnect(controller, false);
679fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(controller, true);
680fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
681fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
682fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
683fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_releaseSession() throws InterruptedException {
684fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
685fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.close();
686fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(mController, true);
687fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
688fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
689fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
690fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_close() throws InterruptedException {
691fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
692fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
693fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(mController, true);
694fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
695fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
696fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
697fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testFastForward() throws InterruptedException {
698fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
699fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
700fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
701fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
702fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onFastForward(MediaSession2 session, ControllerInfo controller) {
703fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
704fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
705fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
706fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
707fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
708fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
709fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
710fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testFastForward").build()) {
711fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
712fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.fastForward();
713fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
714fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
715fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
716fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
717fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
718fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testRewind() throws InterruptedException {
719fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
720fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
721fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
722fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
723fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onRewind(MediaSession2 session, ControllerInfo controller) {
724fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
725fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
726fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
727fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
728fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
729fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
730fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
731fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testRewind").build()) {
732fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
733fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.rewind();
734fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
735fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
736fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
737fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
738fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
739fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPlayFromSearch() throws InterruptedException {
740fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
741fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String request = "random query";
742fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
743fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
744fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
745fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
746fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
747fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlayFromSearch(MediaSession2 session, ControllerInfo controller,
748fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String query, Bundle extras) {
749fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                super.onPlayFromSearch(session, controller, query, extras);
750fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
751fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, query);
752fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
753fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
754fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
755fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
756fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
757fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
758fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
759fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPlayFromSearch").build()) {
760fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
761fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.playFromSearch(request, bundle);
762fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
763fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
764fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
765fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
766fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
767fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPlayFromUri() throws InterruptedException {
768fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
769fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Uri request = Uri.parse("foo://boo");
770fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
771fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
772fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
773fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
774fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
775fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlayFromUri(MediaSession2 session, ControllerInfo controller, Uri uri,
776fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    Bundle extras) {
777fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
778fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, uri);
779fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
780fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
781fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
782fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
783fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
784fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
785fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
786fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPlayFromUri").build()) {
787fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
788fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.playFromUri(request, bundle);
789fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
790fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
791fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
792fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
793fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
794fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPlayFromMediaId() throws InterruptedException {
795fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
796fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String request = "media_id";
797fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
798fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
799fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
800fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
801fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
802fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlayFromMediaId(MediaSession2 session, ControllerInfo controller,
803fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String mediaId, Bundle extras) {
804fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
805fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, mediaId);
806fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
807fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
808fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
809fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
810fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
811fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
812fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
813fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPlayFromMediaId").build()) {
814fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
815fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.playFromMediaId(request, bundle);
816fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
817fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
818fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
819fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
820fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
821fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPrepareFromSearch() throws InterruptedException {
822fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
823fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String request = "random query";
824fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
825fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
826fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
827fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
828fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
829fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPrepareFromSearch(MediaSession2 session, ControllerInfo controller,
830fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String query, Bundle extras) {
831fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
832fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, query);
833fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
834fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
835fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
836fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
837fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
838fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
839fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
840fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPrepareFromSearch").build()) {
841fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
842fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.prepareFromSearch(request, bundle);
843fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
844fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
845fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
846fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
847fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
848fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPrepareFromUri() throws InterruptedException {
849fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
850fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Uri request = Uri.parse("foo://boo");
851fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
852fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
853fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
854fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
855fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
856fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPrepareFromUri(MediaSession2 session, ControllerInfo controller, Uri uri,
857fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    Bundle extras) {
858fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
859fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, uri);
860fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
861fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
862fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
863fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
864fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
865fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
866fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
867fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPrepareFromUri").build()) {
868fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
869fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.prepareFromUri(request, bundle);
870fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
871fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
872fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
873fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
874fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
875fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPrepareFromMediaId() throws InterruptedException {
876fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
877fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String request = "media_id";
878fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
879fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
880fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
881fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
882fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
883fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPrepareFromMediaId(MediaSession2 session, ControllerInfo controller,
884fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String mediaId, Bundle extras) {
885fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
886fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, mediaId);
887fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
888fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
889fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
890fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
891fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
892fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
893fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
894fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPrepareFromMediaId").build()) {
895fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
896fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.prepareFromMediaId(request, bundle);
897fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
898fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
899fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
900fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
901fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
902fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetRating() throws InterruptedException {
903fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
904fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int ratingType = Rating2.RATING_5_STARS;
905fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final float ratingValue = 3.5f;
906fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Rating2 rating = Rating2.newStarRating(ratingType, ratingValue);
907fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String mediaId = "media_id";
908fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
909fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
910fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
911fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
912fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onSetRating(MediaSession2 session, ControllerInfo controller,
913fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String mediaIdOut, Rating2 ratingOut) {
914fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
915fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mediaId, mediaIdOut);
916fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(rating, ratingOut);
917fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
918fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
919fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
920fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
921fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
922fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
923fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
924fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testSetRating").build()) {
925fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
926fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.setRating(mediaId, rating);
927fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
928fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
929fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
930fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
931fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
932fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testIsConnected() throws InterruptedException {
933fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
934fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mController.isConnected());
935fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        sHandler.postAndSync(new Runnable() {
936fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
937fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void run() {
938fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession.close();
939fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
940fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
941fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(mController, true);
942fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertFalse(mController.isConnected());
943fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
944fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
945fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
946fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * Test potential deadlock for calls between controller and session.
947fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
948fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
949fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testDeadlock() throws InterruptedException {
950fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
951fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        sHandler.postAndSync(new Runnable() {
952fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
953fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void run() {
954fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession.close();
955fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession = null;
956fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
957fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
958fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
959fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Two more threads are needed not to block test thread nor test wide thread (sHandler).
960fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final HandlerThread sessionThread = new HandlerThread("testDeadlock_session");
961fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final HandlerThread testThread = new HandlerThread("testDeadlock_test");
962fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        sessionThread.start();
963fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testThread.start();
964fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SyncHandler sessionHandler = new SyncHandler(sessionThread.getLooper());
965fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Handler testHandler = new Handler(testThread.getLooper());
966fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
967fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
968fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            final MockPlayer player = new MockPlayer(0);
969fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            sessionHandler.postAndSync(new Runnable() {
970fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                @Override
971fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                public void run() {
972fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    mSession = new MediaSession2.Builder(mContext)
973fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            .setPlayer(mPlayer)
974fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            .setSessionCallback(sHandlerExecutor, new SessionCallback() {})
975fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            .setId("testDeadlock").build();
976fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
977fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            });
978fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            final MediaController2 controller = createController(mSession.getToken());
979fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            testHandler.post(new Runnable() {
980fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                @Override
981fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                public void run() {
982fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    final int state = MediaPlayerBase.PLAYER_STATE_ERROR;
983fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    for (int i = 0; i < 100; i++) {
984fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        // triggers call from session to controller.
985fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
986fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        // triggers call from controller to session.
987fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.play();
988fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
989fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        // Repeat above
990fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
991fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.pause();
992fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
993fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.reset();
994fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
995fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.skipToNextItem();
996fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
997fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.skipToPreviousItem();
998fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
999fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    // This may hang if deadlock happens.
1000fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    latch.countDown();
1001fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
1002fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            });
1003fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
1004fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } finally {
1005fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            if (mSession != null) {
1006fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                sessionHandler.postAndSync(new Runnable() {
1007fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    @Override
1008fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    public void run() {
1009fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        // Clean up here because sessionHandler will be removed afterwards.
1010fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        mSession.close();
1011fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        mSession = null;
1012fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
1013fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                });
1014fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1015fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            if (sessionThread != null) {
1016fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                sessionThread.quitSafely();
1017fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1018fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            if (testThread != null) {
1019fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                testThread.quitSafely();
1020fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1021fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
1022fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1023fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1024fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1025fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetServiceToken() {
1026fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1027fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        SessionToken2 token = TestUtils.getServiceToken(mContext, MockMediaSessionService2.ID);
1028fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(token);
1029fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(mContext.getPackageName(), token.getPackageName());
1030fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(MockMediaSessionService2.ID, token.getId());
1031fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(SessionToken2.TYPE_SESSION_SERVICE, token.getType());
1032fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1033fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1034fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1035fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testConnectToService_sessionService() throws InterruptedException {
1036fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1037fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testConnectToService(MockMediaSessionService2.ID);
1038fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1039fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1040fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1041fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testConnectToService_libraryService() throws InterruptedException {
1042fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1043fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testConnectToService(MockMediaLibraryService2.ID);
1044fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1045fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1046fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testConnectToService(String id) throws InterruptedException {
1047fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1048fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
1049e57a84cc8a3c62a777d4dcf48d1c7f6e58657bf6Jaewan Kim        final MediaLibrarySessionCallback sessionCallback = new MediaLibrarySessionCallback() {
1050fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1051fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public SessionCommandGroup2 onConnect(@NonNull MediaSession2 session,
1052fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    @NonNull ControllerInfo controller) {
1053fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                if (Process.myUid() == controller.getUid()) {
1054fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    if (mSession != null) {
1055fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        mSession.close();
1056fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
1057fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    mSession = session;
1058fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    mPlayer = (MockPlayer) session.getPlayer();
1059fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    assertEquals(mContext.getPackageName(), controller.getPackageName());
1060fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    assertFalse(controller.isTrusted());
1061fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    latch.countDown();
1062fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
1063fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return super.onConnect(session, controller);
1064fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1065fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
1066fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestServiceRegistry.getInstance().setSessionCallback(sessionCallback);
1067fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1068fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController = createController(TestUtils.getServiceToken(mContext, id));
1069fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1070fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1071fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Test command from controller to session service
1072fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO: Re enable when transport control works
1073fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        /*
1074fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.play();
1075fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1076fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mPlayCalled);
1077fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        */
1078fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1079fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Test command from session service to controller
1080fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO(jaewan): Add equivalent tests again
1081fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        /*
1082fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
1083fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.registerPlayerEventCallback((state) -> {
1084fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertNotNull(state);
1085fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertEquals(PlaybackState.STATE_REWINDING, state.getState());
1086fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            latch.countDown();
1087fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }, sHandler);
1088fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mPlayer.notifyPlaybackState(
1089fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                TestUtils.createPlaybackState(PlaybackState.STATE_REWINDING));
1090fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
1091fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        */
1092fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1093fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1094fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1095fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerAfterSessionIsGone_session() throws InterruptedException {
1096fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1097fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testControllerAfterSessionIsClosed(mSession.getToken().getId());
1098fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1099fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1100fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // TODO(jaewan): Re-enable this test
1101fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Ignore
1102fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1103fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerAfterSessionIsClosed_sessionService() throws InterruptedException {
1104fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1105fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        /*
1106fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        connectToService(TestUtils.getServiceToken(mContext, MockMediaSessionService2.ID));
1107fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testControllerAfterSessionIsClosed(MockMediaSessionService2.ID);
1108fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        */
1109fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1110fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1111fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1112fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    public void testSubscribeRouteInfo() throws InterruptedException {
1113fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        prepareLooper();
1114fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final TestSessionCallback callback = new TestSessionCallback() {
1115fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            @Override
1116fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            public void onSubscribeRoutesInfo(@NonNull MediaSession2 session,
1117fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                    @NonNull ControllerInfo controller) {
1118fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                assertEquals(mContext.getPackageName(), controller.getPackageName());
1119fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                mLatch.countDown();
1120fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            }
1121fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1122fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            @Override
1123fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            public void onUnsubscribeRoutesInfo(@NonNull MediaSession2 session,
1124fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                    @NonNull ControllerInfo controller) {
1125fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                assertEquals(mContext.getPackageName(), controller.getPackageName());
1126fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                mLatch.countDown();
1127fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            }
1128fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        };
1129fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        mSession.close();
1130fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        mSession = new MediaSession2.Builder(mContext).setPlayer(mPlayer)
1131fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                .setSessionCallback(sHandlerExecutor, callback).setId(TAG).build();
1132fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final MediaController2 controller = createController(mSession.getToken());
1133fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1134fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        callback.resetLatchCount(1);
1135fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        controller.subscribeRoutesInfo();
1136fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        assertTrue(callback.mLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1137fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1138fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        callback.resetLatchCount(1);
1139fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        controller.unsubscribeRoutesInfo();
1140fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        assertTrue(callback.mLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1141fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    }
1142fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1143fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    @Test
1144fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    public void testSelectRouteInfo() throws InterruptedException {
1145fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        prepareLooper();
1146fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final Bundle testRoute = new Bundle();
1147fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        testRoute.putString("id", "testRoute");
1148fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final TestSessionCallback callback = new TestSessionCallback() {
1149fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            @Override
1150fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            public void onSelectRoute(@NonNull MediaSession2 session,
1151fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                    @NonNull ControllerInfo controller, @NonNull Bundle route) {
1152fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                assertEquals(mContext.getPackageName(), controller.getPackageName());
1153fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                assertTrue(TestUtils.equals(route, testRoute));
1154fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                mLatch.countDown();
1155fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            }
1156fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        };
1157fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        mSession.close();
1158fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        mSession = new MediaSession2.Builder(mContext).setPlayer(mPlayer)
1159fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                .setSessionCallback(sHandlerExecutor, callback).setId(TAG).build();
1160fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final MediaController2 controller = createController(mSession.getToken());
1161fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1162fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        callback.resetLatchCount(1);
1163fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        controller.selectRoute(testRoute);
1164fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        assertTrue(callback.mLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1165fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    }
1166fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1167fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    @Test
1168fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_beforeConnected() throws InterruptedException {
1169fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1170fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller =
1171fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                createController(mSession.getToken(), false, null);
1172fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.close();
1173fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1174fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1175fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1176fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_twice() {
1177fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1178fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
1179fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
1180fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1181fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1182fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1183fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_session() throws InterruptedException {
1184fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1185fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String id = mSession.getToken().getId();
1186fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
1187fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // close is done immediately for session.
1188fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testNoInteraction();
1189fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1190fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Test whether the controller is notified about later close of the session or
1191fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // re-creation.
1192fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testControllerAfterSessionIsClosed(id);
1193fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1194fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1195fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Ignore
1196fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1197fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_sessionService() throws InterruptedException {
1198fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1199fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testCloseFromService(MockMediaSessionService2.ID);
1200fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1201fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1202fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Ignore
1203fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1204fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_libraryService() throws InterruptedException {
1205fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1206fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testCloseFromService(MockMediaLibraryService2.ID);
1207fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1208fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1209fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    private void testCloseFromService(String id) throws InterruptedException {
1210fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
1211fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestServiceRegistry.getInstance().setSessionServiceCallback(new SessionServiceCallback() {
1212fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1213fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onCreated() {
1214fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                // Do nothing.
1215fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1216fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1217fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1218fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onDestroyed() {
1219fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
1220fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1221fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
1222fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController = createController(TestUtils.getServiceToken(mContext, id));
1223fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
1224fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Wait until close triggers onDestroy() of the session service.
1225fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
1226fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNull(TestServiceRegistry.getInstance().getServiceInstance());
1227fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testNoInteraction();
1228fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1229fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Test whether the controller is notified about later close of the session or
1230fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // re-creation.
1231fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testControllerAfterSessionIsClosed(id);
1232fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1233fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1234fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    private void testControllerAfterSessionIsClosed(final String id) throws InterruptedException {
1235fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // This cause session service to be died.
1236fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.close();
1237fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(mController, true);
1238fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testNoInteraction();
1239fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1240fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Ensure that the controller cannot use newly create session with the same ID.
1241fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Recreated session has different session stub, so previously created controller
1242fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // shouldn't be available.
1243fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession = new MediaSession2.Builder(mContext)
1244fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
1245fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, new SessionCallback() {})
1246fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId(id).build();
1247fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testNoInteraction();
1248fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1249fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1250fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // Test that mSession and mController doesn't interact.
1251fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // Note that this method can be called after the mSession is died, so mSession may not have
1252fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // valid player.
1253fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    private void testNoInteraction() throws InterruptedException {
1254fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO: check that calls from the controller to session shouldn't be delivered.
1255fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1256fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Calls from the session to controller shouldn't be delivered.
1257fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
1258fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        setRunnableForOnCustomCommand(mController, new Runnable() {
1259fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1260fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void run() {
1261fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
1262fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1263fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
1264fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        SessionCommand2 customCommand = new SessionCommand2("testNoInteraction", null);
1265fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.sendCustomCommand(customCommand, null);
1266fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertFalse(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
1267fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        setRunnableForOnCustomCommand(mController, null);
1268fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1269fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1270fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // TODO(jaewan): Add  test for service connect rejection, when we differentiate session
1271fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    //               active/inactive and connection accept/refuse
1272fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1273fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    class TestVolumeProvider extends VolumeProviderCompat {
1274fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch mLatch = new CountDownLatch(1);
1275fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        boolean mSetVolumeToCalled;
1276fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        boolean mAdjustVolumeCalled;
1277fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        int mVolume;
1278fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        int mDirection;
1279fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1280fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestVolumeProvider(int controlType, int maxVolume, int currentVolume) {
1281fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            super(controlType, maxVolume, currentVolume);
1282fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
1283fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1284fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        @Override
1285fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        public void onSetVolumeTo(int volume) {
1286fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mSetVolumeToCalled = true;
1287fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mVolume = volume;
1288fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mLatch.countDown();
1289fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
1290fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1291fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        @Override
1292fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        public void onAdjustVolume(int direction) {
1293fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mAdjustVolumeCalled = true;
1294fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mDirection = direction;
1295fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mLatch.countDown();
1296fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
1297fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1298fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1299fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    class TestSessionCallback extends SessionCallback {
1300fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        CountDownLatch mLatch;
1301fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1302fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        void resetLatchCount(int count) {
1303fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            mLatch = new CountDownLatch(count);
1304fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        }
1305fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    }
1306fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang}
1307