MediaController2Test.java revision 1a125123e09a1f0e2991b1fce907970d8993667e
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.Test;
53fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport org.junit.runner.RunWith;
54fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
55fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.lang.reflect.Method;
56fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.util.List;
57fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.util.concurrent.CountDownLatch;
58fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.util.concurrent.TimeUnit;
59fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangimport java.util.concurrent.atomic.AtomicReference;
60fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
61fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang/**
62fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang * Tests {@link MediaController2}.
63fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang */
64fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang// TODO(jaewan): Implement host-side test so controller and session can run in different processes.
65fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang// TODO(jaewan): Fix flaky failure -- see MediaController2Impl.getController()
66fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang// TODO(jaeawn): Revisit create/close session in the sHandler. It's no longer necessary.
67fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang@SdkSuppress(minSdkVersion = Build.VERSION_CODES.KITKAT)
68fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang@RunWith(AndroidJUnit4.class)
69fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang@SmallTest
70fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang@FlakyTest
71fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kangpublic class MediaController2Test extends MediaSession2TestBase {
72fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    private static final String TAG = "MediaController2Test";
73fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
74fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    PendingIntent mIntent;
75fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    MediaSession2 mSession;
76fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    MediaController2 mController;
77fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    MockPlayer mPlayer;
78fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    MockPlaylistAgent mMockAgent;
79fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
80fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Before
81fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Override
82fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void setUp() throws Exception {
83fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        super.setUp();
84fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Intent sessionActivity = new Intent(mContext, MockActivity.class);
85fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Create this test specific MediaSession2 to use our own Handler.
86fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mIntent = PendingIntent.getActivity(mContext, 0, sessionActivity, 0);
87fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
88fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mPlayer = new MockPlayer(1);
89fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mMockAgent = new MockPlaylistAgent();
90fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession = new MediaSession2.Builder(mContext)
91fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
92fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlaylistAgent(mMockAgent)
93fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, new SessionCallback() {
94fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    @Override
95fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    public SessionCommandGroup2 onConnect(MediaSession2 session,
96fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            ControllerInfo controller) {
97fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        if (Process.myUid() == controller.getUid()) {
98fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            return super.onConnect(session, controller);
99fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        }
100fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        return null;
101fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
102fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
103fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    @Override
104fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    public void onPlaylistMetadataChanged(MediaSession2 session,
105fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            MediaPlaylistAgent playlistAgent,
106fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            MediaMetadata2 metadata) {
107fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        super.onPlaylistMetadataChanged(session, playlistAgent, metadata);
108fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
109fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                })
110fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionActivity(mIntent)
111fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId(TAG).build();
112fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController = createController(mSession.getToken());
113fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestServiceRegistry.getInstance().setHandler(sHandler);
114fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
115fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
116fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @After
117fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Override
118fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void cleanUp() throws Exception {
119fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        super.cleanUp();
120fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        if (mSession != null) {
121fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mSession.close();
122fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
123fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestServiceRegistry.getInstance().cleanUp();
124fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
125fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
126fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
127fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * Test if the {@link MediaSession2TestBase.TestControllerCallback} wraps the callback proxy
128fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * without missing any method.
129fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
130fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
131fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testTestControllerCallback() {
132fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
133fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        Method[] methods = TestControllerCallback.class.getMethods();
134fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(methods);
135fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        for (int i = 0; i < methods.length; i++) {
136fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            // For any methods in the controller callback, TestControllerCallback should have
137fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            // overriden the method and call matching API in the callback proxy.
138fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertNotEquals("TestControllerCallback should override " + methods[i]
139fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            + " and call callback proxy",
140fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    ControllerCallback.class, methods[i].getDeclaringClass());
141fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
142fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
143fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
144fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
145fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPlay() {
146fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
147fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.play();
148fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
149fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
150fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
151fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
152fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
153fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mPlayCalled);
154fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
155fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
156fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
157fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPause() {
158fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
159fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.pause();
160fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
161fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
162fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
163fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
164fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
165fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mPauseCalled);
166fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
167fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
168fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
169fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testReset() {
170fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
171fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.reset();
172fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
173fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
174fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
175fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
176fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
177fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mResetCalled);
178fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
179fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
180fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
181fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPrepare() {
182fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
183fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.prepare();
184fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
185fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
186fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
187fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
188fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
189fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mPrepareCalled);
190fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
191fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
192fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
193fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSeekTo() {
194fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
195fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final long seekPosition = 12125L;
196fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.seekTo(seekPosition);
197fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
198fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(mPlayer.mCountDownLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
199fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } catch (InterruptedException e) {
200fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            fail(e.getMessage());
201fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
202fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mSeekToCalled);
203fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(seekPosition, mPlayer.mSeekPosition);
204fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
205fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
206fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
207fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGettersAfterConnected() throws InterruptedException {
208fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
209fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int state = MediaPlayerBase.PLAYER_STATE_PLAYING;
210ebc904f04b5f4c011ec791b1b0e94660ac02fdbaHyundo Moon        final int bufferingState = MediaPlayerBase.BUFFERING_STATE_BUFFERING_COMPLETE;
211fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final long position = 150000;
212fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final long bufferedPosition = 900000;
213fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final float speed = 0.5f;
2141a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final long timeDiff = 102;
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
224fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken());
2251a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        controller.setTimeDiff(timeDiff);
226fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(state, controller.getPlayerState());
227fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(bufferedPosition, controller.getBufferedPosition());
228e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        assertEquals(speed, controller.getPlaybackSpeed(), 0.0f);
2291a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        assertEquals(position + (long) (speed * timeDiff), controller.getCurrentPosition());
2303a151f1283d08b40c12e4a17903ba421f23c4342Hyundo Moon        assertEquals(currentMediaItem, controller.getCurrentMediaItem());
231fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
232fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
233fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
234fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetSessionActivity() {
235fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
236fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        PendingIntent sessionActivity = mController.getSessionActivity();
237fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(mContext.getPackageName(), sessionActivity.getCreatorPackage());
238fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(Process.myUid(), sessionActivity.getCreatorUid());
239fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
240fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
241fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
242fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetPlaylist() throws InterruptedException {
243fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
244fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final List<MediaItem2> list = TestUtils.createPlaylist(2);
245fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.setPlaylist(list, null /* Metadata */);
246fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
247fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
248fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSetPlaylistCalled);
249fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNull(mMockAgent.mMetadata);
250fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
251fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(mMockAgent.mPlaylist);
252fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(list.size(), mMockAgent.mPlaylist.size());
253fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        for (int i = 0; i < list.size(); i++) {
254fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            // MediaController2.setPlaylist does not ensure the equality of the items.
255fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertEquals(list.get(i).getMediaId(), mMockAgent.mPlaylist.get(i).getMediaId());
256fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
257fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
258fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
259fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
260fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * This also tests {@link ControllerCallback#onPlaylistChanged(
261fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * MediaController2, List, MediaMetadata2)}.
262fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
263fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
264fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetPlaylist() throws InterruptedException {
265fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
266fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final List<MediaItem2> testList = TestUtils.createPlaylist(2);
267fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final AtomicReference<List<MediaItem2>> listFromCallback = new AtomicReference<>();
268fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
269fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
270fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
271fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlaylistChanged(MediaController2 controller,
272fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    List<MediaItem2> playlist, MediaMetadata2 metadata) {
273fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertNotNull(playlist);
274fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testList.size(), playlist.size());
275fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                for (int i = 0; i < playlist.size(); i++) {
276fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    assertEquals(testList.get(i).getMediaId(), playlist.get(i).getMediaId());
277fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
278fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                listFromCallback.set(playlist);
279fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
280fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
281fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
282fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
283fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
284fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public List<MediaItem2> getPlaylist() {
285fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return testList;
286fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
287fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
288fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
289fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
290fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testControllerCallback_onPlaylistChanged")
291fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, new SessionCallback() {})
292fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlaylistAgent(agent)
293fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .build()) {
294fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(
295fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    session.getToken(), true, callback);
296fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            agent.notifyPlaylistChanged();
297fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
298fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertEquals(listFromCallback.get(), controller.getPlaylist());
299fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
300fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
301fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
302fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
303fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testUpdatePlaylistMetadata() throws InterruptedException {
304fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
305fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaMetadata2 testMetadata = TestUtils.createMetadata();
306fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.updatePlaylistMetadata(testMetadata);
307fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
308fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
309fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mUpdatePlaylistMetadataCalled);
310fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(mMockAgent.mMetadata);
311fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testMetadata.getMediaId(), mMockAgent.mMetadata.getMediaId());
312fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
313fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
314fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
315fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetPlaylistMetadata() throws InterruptedException {
316fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
317fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaMetadata2 testMetadata = TestUtils.createMetadata();
318fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final AtomicReference<MediaMetadata2> metadataFromCallback = new AtomicReference<>();
319fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
320fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
321fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
322fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlaylistMetadataChanged(MediaController2 controller,
323fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    MediaMetadata2 metadata) {
324fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertNotNull(testMetadata);
325fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testMetadata.getMediaId(), metadata.getMediaId());
326fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                metadataFromCallback.set(metadata);
327fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
328fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
329fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
330fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
331fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
332fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public MediaMetadata2 getPlaylistMetadata() {
333fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return testMetadata;
334fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
335fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
336fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
337fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
338fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testGetPlaylistMetadata")
339fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, new SessionCallback() {})
340fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlaylistAgent(agent)
341fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .build()) {
342fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken(), true, callback);
343fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            agent.notifyPlaylistMetadataChanged();
344fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
345fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertEquals(metadataFromCallback.get().getMediaId(),
346fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    controller.getPlaylistMetadata().getMediaId());
347fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
348fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
349fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
350e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon    @Test
351e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon    public void testSetPlaybackSpeed() throws Exception {
352e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        prepareLooper();
353e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        final float speed = 1.5f;
354e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        mController.setPlaybackSpeed(speed);
355e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        assertTrue(mPlayer.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
356e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon        assertEquals(speed, mPlayer.mPlaybackSpeed, 0.0f);
357e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon    }
358e18fafc56c49f683278d91e16a7246c7b1e01badHyundo Moon
359fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
360fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * Test whether {@link MediaSession2#setPlaylist(List, MediaMetadata2)} is notified
361fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * through the
362fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * {@link ControllerCallback#onPlaylistMetadataChanged(MediaController2, MediaMetadata2)}
363fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * if the controller doesn't have {@link SessionCommand2#COMMAND_CODE_PLAYLIST_GET_LIST} but
364fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * {@link SessionCommand2#COMMAND_CODE_PLAYLIST_GET_LIST_METADATA}.
365fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
366fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
367fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_onPlaylistMetadataChanged() throws InterruptedException {
368fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
369fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaItem2 item = TestUtils.createMediaItemWithMetadata();
370fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final List<MediaItem2> list = TestUtils.createPlaylist(2);
371fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
372fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
373fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
374fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlaylistMetadataChanged(MediaController2 controller,
375fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    MediaMetadata2 metadata) {
376fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertNotNull(metadata);
377fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(item.getMediaId(), metadata.getMediaId());
378fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
379fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
380fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
381fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback sessionCallback = new SessionCallback() {
382fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
383fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public SessionCommandGroup2 onConnect(MediaSession2 session,
384fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    ControllerInfo controller) {
385fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                if (Process.myUid() == controller.getUid()) {
386fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    SessionCommandGroup2 commands = new SessionCommandGroup2();
387fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    commands.addCommand(new SessionCommand2(
388fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                              SessionCommand2.COMMAND_CODE_PLAYLIST_GET_LIST_METADATA));
389fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    return commands;
390fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
391fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return super.onConnect(session, controller);
392fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
393fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
394fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
395fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
396fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public MediaMetadata2 getPlaylistMetadata() {
397fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return item.getMetadata();
398fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
399fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
400fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
401fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public List<MediaItem2> getPlaylist() {
402fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return list;
403fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
404fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
405fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
406fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
407fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testControllerCallback_onPlaylistMetadataChanged")
408fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, sessionCallback)
409fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlaylistAgent(agent)
410fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .build()) {
411fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken(), true, callback);
412fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            agent.notifyPlaylistMetadataChanged();
413fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
414fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
415fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
416fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
4171a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim
4181a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    @Test
4191a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    public void testControllerCallback_onSeekCompleted() throws InterruptedException {
4201a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        prepareLooper();
4211a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final long testSeekPosition = 400;
4221a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final long testPosition = 500;
4231a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final CountDownLatch latch = new CountDownLatch(1);
4241a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final ControllerCallback callback = new ControllerCallback() {
4251a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            @Override
4261a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            public void onSeekCompleted(MediaController2 controller, long position) {
4271a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                controller.setTimeDiff(Long.valueOf(0));
4281a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testSeekPosition, position);
4291a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testPosition, controller.getCurrentPosition());
4301a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                latch.countDown();
4311a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            }
4321a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        };
4331a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final MediaController2 controller = createController(mSession.getToken(), true, callback);
4341a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        mPlayer.mCurrentPosition = testPosition;
4351a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        mPlayer.notifySeekCompleted(testSeekPosition);
4361a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
4371a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    }
4381a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim
4391a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    @Test
4401a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    public void testControllerCallback_onBufferingStateChanged() throws InterruptedException {
4411a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        prepareLooper();
4421a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final List<MediaItem2> testPlaylist = TestUtils.createPlaylist(3);
4431a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final MediaItem2 testItem = testPlaylist.get(0);
4441a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final int testBufferingState = MediaPlayerBase.BUFFERING_STATE_BUFFERING_AND_PLAYABLE;
4451a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final long testBufferingPosition = 500;
4461a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final CountDownLatch latch = new CountDownLatch(1);
4471a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final ControllerCallback callback = new ControllerCallback() {
4481a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            @Override
4491a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            public void onBufferingStateChanged(MediaController2 controller, MediaItem2 item,
4501a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                    int state) {
4511a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                controller.setTimeDiff(Long.valueOf(0));
4521a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testItem, item);
4531a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testBufferingState, state);
4541a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testBufferingState, controller.getBufferingState());
4551a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testBufferingPosition, controller.getBufferedPosition());
4561a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                latch.countDown();
4571a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            }
4581a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        };
4591a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final MediaController2 controller = createController(mSession.getToken(), true, callback);
4601a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        mSession.setPlaylist(testPlaylist, null);
4611a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        mPlayer.mBufferedPosition = testBufferingPosition;
4621a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        mPlayer.notifyBufferingStateChanged(testItem.getDataSourceDesc(), testBufferingState);
4631a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
4641a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    }
4651a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim
4661a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    @Test
4671a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    public void testControllerCallback_onPlayerStateChanged() throws InterruptedException {
4681a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        prepareLooper();
4691a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final int testPlayerState = MediaPlayerBase.PLAYER_STATE_PLAYING;
4701a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final long testPosition = 500;
4711a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final CountDownLatch latch = new CountDownLatch(1);
4721a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final ControllerCallback callback = new ControllerCallback() {
4731a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            @Override
4741a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            public void onPlayerStateChanged(MediaController2 controller, int state) {
4751a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                controller.setTimeDiff(Long.valueOf(0));
4761a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testPlayerState, state);
4771a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testPlayerState, controller.getPlayerState());
4781a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                assertEquals(testPosition, controller.getCurrentPosition());
4791a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim                latch.countDown();
4801a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim            }
4811a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        };
4821a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        final MediaController2 controller = createController(mSession.getToken(), true, callback);
4831a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        mPlayer.mCurrentPosition = testPosition;
4841a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        mPlayer.notifyPlaybackState(testPlayerState);
4851a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
4861a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim    }
4871a125123e09a1f0e2991b1fce907970d8993667eJaewan Kim
488fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
489fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testAddPlaylistItem() throws InterruptedException {
490fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
491fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testIndex = 12;
492fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaItem2 testMediaItem = TestUtils.createMediaItemWithMetadata();
493fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.addPlaylistItem(testIndex, testMediaItem);
494fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
495fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
496fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mAddPlaylistItemCalled);
497fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testIndex, mMockAgent.mIndex);
498fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // MediaController2.addPlaylistItem does not ensure the equality of the items.
499fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testMediaItem.getMediaId(), mMockAgent.mItem.getMediaId());
500fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
501fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
502fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
503fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testRemovePlaylistItem() throws InterruptedException {
504fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
505fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mMockAgent.mPlaylist = TestUtils.createPlaylist(2);
506fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
507fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Recreate controller for sending removePlaylistItem.
508fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // It's easier to ensure that MediaController2.getPlaylist() returns the playlist from the
509fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // agent.
510fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken());
511fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaItem2 targetItem = controller.getPlaylist().get(0);
512fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.removePlaylistItem(targetItem);
513fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
514fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
515fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mRemovePlaylistItemCalled);
516fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(targetItem, mMockAgent.mItem);
517fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
518fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
519fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
520fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testReplacePlaylistItem() throws InterruptedException {
521fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
522fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testIndex = 12;
523fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaItem2 testMediaItem = TestUtils.createMediaItemWithMetadata();
524fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.replacePlaylistItem(testIndex, testMediaItem);
525fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
526fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
527fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mReplacePlaylistItemCalled);
528fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // MediaController2.replacePlaylistItem does not ensure the equality of the items.
529fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testMediaItem.getMediaId(), mMockAgent.mItem.getMediaId());
530fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
531fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
532fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
533fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSkipToPreviousItem() throws InterruptedException {
534fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
535fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.skipToPreviousItem();
536fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
537fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSkipToPreviousItemCalled);
538fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
539fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
540fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
541fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSkipToNextItem() throws InterruptedException {
542fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
543fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.skipToNextItem();
544fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
545fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSkipToNextItemCalled);
546fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
547fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
548fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
549fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSkipToPlaylistItem() throws InterruptedException {
550fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
551fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken());
552fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaItem2 targetItem = TestUtils.createMediaItemWithMetadata();
553fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.skipToPlaylistItem(targetItem);
554fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
555fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
556fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSkipToPlaylistItemCalled);
557fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(targetItem, mMockAgent.mItem);
558fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
559fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
560fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
561fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * This also tests {@link ControllerCallback#onShuffleModeChanged(MediaController2, int)}.
562fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
563fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
564fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetShuffleMode() throws InterruptedException {
565fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
566fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testShuffleMode = MediaPlaylistAgent.SHUFFLE_MODE_GROUP;
567fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
568fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
569fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public int getShuffleMode() {
570fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return testShuffleMode;
571fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
572fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
573fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
574fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
575fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
576fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onShuffleModeChanged(MediaController2 controller, int shuffleMode) {
577fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testShuffleMode, shuffleMode);
578fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
579fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
580fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
581fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.updatePlayer(mPlayer, agent, null);
582fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken(), true, callback);
583fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        agent.notifyShuffleModeChanged();
584fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
585fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testShuffleMode, controller.getShuffleMode());
586fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
587fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
588fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
589fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetShuffleMode() throws InterruptedException {
590fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
591fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testShuffleMode = MediaPlaylistAgent.SHUFFLE_MODE_GROUP;
592fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.setShuffleMode(testShuffleMode);
593fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
594fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
595fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSetShuffleModeCalled);
596fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testShuffleMode, mMockAgent.mShuffleMode);
597fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
598fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
599fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
600fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * This also tests {@link ControllerCallback#onRepeatModeChanged(MediaController2, int)}.
601fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
602fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
603fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetRepeatMode() throws InterruptedException {
604fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
605fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testRepeatMode = MediaPlaylistAgent.REPEAT_MODE_GROUP;
606fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaPlaylistAgent agent = new MockPlaylistAgent() {
607fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
608fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public int getRepeatMode() {
609fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return testRepeatMode;
610fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
611fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
612fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
613fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final ControllerCallback callback = new ControllerCallback() {
614fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
615fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onRepeatModeChanged(MediaController2 controller, int repeatMode) {
616fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testRepeatMode, repeatMode);
617fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
618fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
619fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
620fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.updatePlayer(mPlayer, agent, null);
621fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken(), true, callback);
622fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        agent.notifyRepeatModeChanged();
623fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
624fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testRepeatMode, controller.getRepeatMode());
625fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
626fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
627fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
628fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetRepeatMode() throws InterruptedException {
629fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
630fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int testRepeatMode = MediaPlaylistAgent.REPEAT_MODE_GROUP;
631fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.setRepeatMode(testRepeatMode);
632fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
633fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
634fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mMockAgent.mSetRepeatModeCalled);
635fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(testRepeatMode, mMockAgent.mRepeatMode);
636fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
637fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
638fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
639fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetVolumeTo() throws Exception {
640fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO(jaewan): Also test with local volume.
641fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
642fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int maxVolume = 100;
643fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int currentVolume = 23;
644fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int volumeControlType = VolumeProviderCompat.VOLUME_CONTROL_ABSOLUTE;
645fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestVolumeProvider volumeProvider =
646fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                new TestVolumeProvider(volumeControlType, maxVolume, currentVolume);
647fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
648fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.updatePlayer(new MockPlayer(0), null, volumeProvider);
649fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaController2 controller = createController(mSession.getToken(), true, null);
650fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
651fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int targetVolume = 50;
652fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.setVolumeTo(targetVolume, 0 /* flags */);
653fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(volumeProvider.mLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
654fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(volumeProvider.mSetVolumeToCalled);
655fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(targetVolume, volumeProvider.mVolume);
656fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
657fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
658fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
659fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testAdjustVolume() throws Exception {
660fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO(jaewan): Also test with local volume.
661fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
662fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int maxVolume = 100;
663fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int currentVolume = 23;
664fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int volumeControlType = VolumeProviderCompat.VOLUME_CONTROL_ABSOLUTE;
665fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestVolumeProvider volumeProvider =
666fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                new TestVolumeProvider(volumeControlType, maxVolume, currentVolume);
667fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
668fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.updatePlayer(new MockPlayer(0), null, volumeProvider);
669fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaController2 controller = createController(mSession.getToken(), true, null);
670fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
671fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int direction = AudioManager.ADJUST_RAISE;
672fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.adjustVolume(direction, 0 /* flags */);
673fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(volumeProvider.mLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
674fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(volumeProvider.mAdjustVolumeCalled);
675fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(direction, volumeProvider.mDirection);
676fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
677fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
678fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
679fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetPackageName() {
680fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
681fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(mContext.getPackageName(), mController.getSessionToken().getPackageName());
682fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
683fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
684fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
685fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSendCustomCommand() throws InterruptedException {
686fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
687fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO(jaewan): Need to revisit with the permission.
688fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCommand2 testCommand =
689fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                new SessionCommand2(SessionCommand2.COMMAND_CODE_PLAYBACK_PREPARE);
690fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle testArgs = new Bundle();
691fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testArgs.putString("args", "testSendCustomCommand");
692fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
693fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
694fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
695fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
696fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onCustomCommand(MediaSession2 session, ControllerInfo controller,
697fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    SessionCommand2 customCommand, Bundle args, ResultReceiver cb) {
698fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
699fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(testCommand, customCommand);
700fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(testArgs, args));
701fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertNull(cb);
702fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
703fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
704fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
705fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.close();
706fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession = new MediaSession2.Builder(mContext).setPlayer(mPlayer)
707fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback).setId(TAG).build();
708fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaController2 controller = createController(mSession.getToken());
709fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.sendCustomCommand(testCommand, testArgs, null);
710fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
711fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
712fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
713fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
714fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_onConnected() throws InterruptedException {
715fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
716fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // createController() uses controller callback to wait until the controller becomes
717fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // available.
718fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller = createController(mSession.getToken());
719fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(controller);
720fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
721fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
722fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
723fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_sessionRejects() throws InterruptedException {
724fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
725fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final MediaSession2.SessionCallback sessionCallback = new SessionCallback() {
726fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
727fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public SessionCommandGroup2 onConnect(MediaSession2 session,
728fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    ControllerInfo controller) {
729fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return null;
730fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
731fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
732fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        sHandler.postAndSync(new Runnable() {
733fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
734fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void run() {
735fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession.close();
736fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession = new MediaSession2.Builder(mContext).setPlayer(mPlayer)
737fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        .setSessionCallback(sHandlerExecutor, sessionCallback).build();
738fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
739fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
740fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller =
741fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                createController(mSession.getToken(), false, null);
742fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(controller);
743fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForConnect(controller, false);
744fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(controller, true);
745fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
746fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
747fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
748fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_releaseSession() throws InterruptedException {
749fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
750fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.close();
751fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(mController, true);
752fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
753fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
754fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
755fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerCallback_close() throws InterruptedException {
756fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
757fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
758fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(mController, true);
759fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
760fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
761fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
762fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testFastForward() throws InterruptedException {
763fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
764fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
765fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
766fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
767fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onFastForward(MediaSession2 session, ControllerInfo controller) {
768fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
769fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
770fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
771fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
772fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
773fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
774fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
775fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testFastForward").build()) {
776fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
777fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.fastForward();
778fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
779fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
780fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
781fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
782fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
783fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testRewind() throws InterruptedException {
784fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
785fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
786fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
787fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
788fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onRewind(MediaSession2 session, ControllerInfo controller) {
789fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
790fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
791fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
792fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
793fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
794fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
795fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
796fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testRewind").build()) {
797fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
798fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.rewind();
799fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
800fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
801fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
802fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
803fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
804fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPlayFromSearch() throws InterruptedException {
805fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
806fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String request = "random query";
807fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
808fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
809fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
810fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
811fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
812fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlayFromSearch(MediaSession2 session, ControllerInfo controller,
813fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String query, Bundle extras) {
814fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                super.onPlayFromSearch(session, controller, query, extras);
815fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
816fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, query);
817fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
818fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
819fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
820fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
821fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
822fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
823fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
824fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPlayFromSearch").build()) {
825fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
826fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.playFromSearch(request, bundle);
827fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
828fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
829fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
830fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
831fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
832fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPlayFromUri() throws InterruptedException {
833fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
834fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Uri request = Uri.parse("foo://boo");
835fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
836fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
837fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
838fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
839fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
840fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlayFromUri(MediaSession2 session, ControllerInfo controller, Uri uri,
841fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    Bundle extras) {
842fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
843fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, uri);
844fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
845fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
846fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
847fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
848fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
849fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
850fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
851fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPlayFromUri").build()) {
852fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
853fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.playFromUri(request, bundle);
854fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
855fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
856fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
857fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
858fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
859fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPlayFromMediaId() throws InterruptedException {
860fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
861fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String request = "media_id";
862fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
863fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
864fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
865fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
866fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
867fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPlayFromMediaId(MediaSession2 session, ControllerInfo controller,
868fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String mediaId, Bundle extras) {
869fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
870fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, mediaId);
871fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
872fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
873fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
874fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
875fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
876fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
877fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
878fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPlayFromMediaId").build()) {
879fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
880fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.playFromMediaId(request, bundle);
881fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
882fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
883fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
884fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
885fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
886fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPrepareFromSearch() throws InterruptedException {
887fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
888fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String request = "random query";
889fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
890fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
891fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
892fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
893fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
894fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPrepareFromSearch(MediaSession2 session, ControllerInfo controller,
895fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String query, Bundle extras) {
896fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
897fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, query);
898fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
899fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
900fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
901fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
902fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
903fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
904fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
905fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPrepareFromSearch").build()) {
906fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
907fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.prepareFromSearch(request, bundle);
908fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
909fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
910fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
911fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
912fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
913fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPrepareFromUri() throws InterruptedException {
914fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
915fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Uri request = Uri.parse("foo://boo");
916fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
917fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
918fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
919fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
920fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
921fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPrepareFromUri(MediaSession2 session, ControllerInfo controller, Uri uri,
922fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    Bundle extras) {
923fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
924fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, uri);
925fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
926fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
927fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
928fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
929fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
930fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
931fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
932fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPrepareFromUri").build()) {
933fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
934fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.prepareFromUri(request, bundle);
935fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
936fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
937fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
938fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
939fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
940fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testPrepareFromMediaId() throws InterruptedException {
941fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
942fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String request = "media_id";
943fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Bundle bundle = new Bundle();
944fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        bundle.putString("key", "value");
945fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
946fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
947fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
948fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onPrepareFromMediaId(MediaSession2 session, ControllerInfo controller,
949fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String mediaId, Bundle extras) {
950fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
951fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(request, mediaId);
952fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertTrue(TestUtils.equals(bundle, extras));
953fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
954fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
955fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
956fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
957fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
958fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
959fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testPrepareFromMediaId").build()) {
960fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
961fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.prepareFromMediaId(request, bundle);
962fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
963fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
964fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
965fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
966fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
967fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testSetRating() throws InterruptedException {
968fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
969fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final int ratingType = Rating2.RATING_5_STARS;
970fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final float ratingValue = 3.5f;
971fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Rating2 rating = Rating2.newStarRating(ratingType, ratingValue);
972fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String mediaId = "media_id";
973fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
974fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
975fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SessionCallback callback = new SessionCallback() {
976fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
977fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onSetRating(MediaSession2 session, ControllerInfo controller,
978fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    String mediaIdOut, Rating2 ratingOut) {
979fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mContext.getPackageName(), controller.getPackageName());
980fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(mediaId, mediaIdOut);
981fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                assertEquals(rating, ratingOut);
982fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
983fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
984fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
985fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
986fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try (MediaSession2 session = new MediaSession2.Builder(mContext)
987fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
988fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, callback)
989fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId("testSetRating").build()) {
990fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            MediaController2 controller = createController(session.getToken());
991fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            controller.setRating(mediaId, rating);
992fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
993fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
994fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
995fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
996fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
997fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testIsConnected() throws InterruptedException {
998fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
999fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mController.isConnected());
1000fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        sHandler.postAndSync(new Runnable() {
1001fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1002fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void run() {
1003fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession.close();
1004fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1005fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
1006fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(mController, true);
1007fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertFalse(mController.isConnected());
1008fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1009fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1010fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    /**
1011fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     * Test potential deadlock for calls between controller and session.
1012fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang     */
1013fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1014fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testDeadlock() throws InterruptedException {
1015fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1016fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        sHandler.postAndSync(new Runnable() {
1017fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1018fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void run() {
1019fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession.close();
1020fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                mSession = null;
1021fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1022fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
1023fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1024fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Two more threads are needed not to block test thread nor test wide thread (sHandler).
1025fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final HandlerThread sessionThread = new HandlerThread("testDeadlock_session");
1026fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final HandlerThread testThread = new HandlerThread("testDeadlock_test");
1027fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        sessionThread.start();
1028fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testThread.start();
1029fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final SyncHandler sessionHandler = new SyncHandler(sessionThread.getLooper());
1030fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final Handler testHandler = new Handler(testThread.getLooper());
1031fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
1032fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        try {
1033fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            final MockPlayer player = new MockPlayer(0);
1034fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            sessionHandler.postAndSync(new Runnable() {
1035fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                @Override
1036fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                public void run() {
1037fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    mSession = new MediaSession2.Builder(mContext)
1038fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            .setPlayer(mPlayer)
1039fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            .setSessionCallback(sHandlerExecutor, new SessionCallback() {})
1040fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                            .setId("testDeadlock").build();
1041fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
1042fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            });
1043fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            final MediaController2 controller = createController(mSession.getToken());
1044fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            testHandler.post(new Runnable() {
1045fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                @Override
1046fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                public void run() {
1047fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    final int state = MediaPlayerBase.PLAYER_STATE_ERROR;
1048fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    for (int i = 0; i < 100; i++) {
1049fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        // triggers call from session to controller.
1050fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
1051fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        // triggers call from controller to session.
1052fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.play();
1053fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1054fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        // Repeat above
1055fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
1056fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.pause();
1057fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
1058fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.reset();
1059fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
1060fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.skipToNextItem();
1061fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        player.notifyPlaybackState(state);
1062fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        controller.skipToPreviousItem();
1063fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
1064fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    // This may hang if deadlock happens.
1065fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    latch.countDown();
1066fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
1067fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            });
1068fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            assertTrue(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
1069fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        } finally {
1070fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            if (mSession != null) {
1071fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                sessionHandler.postAndSync(new Runnable() {
1072fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    @Override
1073fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    public void run() {
1074fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        // Clean up here because sessionHandler will be removed afterwards.
1075fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        mSession.close();
1076fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        mSession = null;
1077fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
1078fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                });
1079fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1080fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            if (sessionThread != null) {
1081fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                sessionThread.quitSafely();
1082fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1083fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            if (testThread != null) {
1084fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                testThread.quitSafely();
1085fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1086fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
1087fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1088fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1089fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1090fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testGetServiceToken() {
1091fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1092fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        SessionToken2 token = TestUtils.getServiceToken(mContext, MockMediaSessionService2.ID);
1093fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNotNull(token);
1094fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(mContext.getPackageName(), token.getPackageName());
1095fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(MockMediaSessionService2.ID, token.getId());
1096fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertEquals(SessionToken2.TYPE_SESSION_SERVICE, token.getType());
1097fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1098fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1099fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1100fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testConnectToService_sessionService() throws InterruptedException {
1101fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1102fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testConnectToService(MockMediaSessionService2.ID);
1103fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1104fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1105fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1106fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testConnectToService_libraryService() throws InterruptedException {
1107fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1108fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testConnectToService(MockMediaLibraryService2.ID);
1109fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1110fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1111fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testConnectToService(String id) throws InterruptedException {
1112fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1113fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
1114e57a84cc8a3c62a777d4dcf48d1c7f6e58657bf6Jaewan Kim        final MediaLibrarySessionCallback sessionCallback = new MediaLibrarySessionCallback() {
1115fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1116fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public SessionCommandGroup2 onConnect(@NonNull MediaSession2 session,
1117fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    @NonNull ControllerInfo controller) {
1118fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                if (Process.myUid() == controller.getUid()) {
1119fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    if (mSession != null) {
1120fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                        mSession.close();
1121fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    }
1122fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    mSession = session;
1123fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    mPlayer = (MockPlayer) session.getPlayer();
1124fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    assertEquals(mContext.getPackageName(), controller.getPackageName());
1125fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    assertFalse(controller.isTrusted());
1126fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                    latch.countDown();
1127fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                }
1128fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                return super.onConnect(session, controller);
1129fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1130fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        };
1131fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestServiceRegistry.getInstance().setSessionCallback(sessionCallback);
1132fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
11338138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim        final SessionCommand2 testCommand = new SessionCommand2("testConnectToService", null);
11348138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim        final CountDownLatch controllerLatch = new CountDownLatch(1);
11358138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim        mController = createController(TestUtils.getServiceToken(mContext, id), true,
11368138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                new ControllerCallback() {
11378138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                    @Override
11388138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                    public void onCustomCommand(MediaController2 controller,
11398138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                            SessionCommand2 command, Bundle args, ResultReceiver receiver) {
11408138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                        if (testCommand.equals(command)) {
11418138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                            controllerLatch.countDown();
11428138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                        }
11438138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                    }
11448138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim                }
11458138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim        );
1146fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1147fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
11488138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim        // Test command from controller to session service.
1149fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.play();
1150fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mCountDownLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1151fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(mPlayer.mPlayCalled);
1152fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
11538138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim        // Test command from session service to controller.
11548138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim        mSession.sendCustomCommand(testCommand, null);
11558138c8606070fc6193f0a9bc566c4b5f1d035457Jaewan Kim        assertTrue(controllerLatch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
1156fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1157fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1158fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1159fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerAfterSessionIsGone_session() throws InterruptedException {
1160fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1161fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testControllerAfterSessionIsClosed(mSession.getToken().getId());
1162fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1163fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1164fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1165fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testControllerAfterSessionIsClosed_sessionService() throws InterruptedException {
1166fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
11679d52d2ed9a9120a066cccaf0a544dec20a52b95aJaewan Kim        testConnectToService(MockMediaSessionService2.ID);
1168fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testControllerAfterSessionIsClosed(MockMediaSessionService2.ID);
1169fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1170fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1171fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1172fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    public void testSubscribeRouteInfo() throws InterruptedException {
1173fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        prepareLooper();
1174fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final TestSessionCallback callback = new TestSessionCallback() {
1175fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            @Override
1176fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            public void onSubscribeRoutesInfo(@NonNull MediaSession2 session,
1177fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                    @NonNull ControllerInfo controller) {
1178fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                assertEquals(mContext.getPackageName(), controller.getPackageName());
1179fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                mLatch.countDown();
1180fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            }
1181fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1182fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            @Override
1183fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            public void onUnsubscribeRoutesInfo(@NonNull MediaSession2 session,
1184fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                    @NonNull ControllerInfo controller) {
1185fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                assertEquals(mContext.getPackageName(), controller.getPackageName());
1186fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                mLatch.countDown();
1187fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            }
1188fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        };
1189fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        mSession.close();
1190fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        mSession = new MediaSession2.Builder(mContext).setPlayer(mPlayer)
1191fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                .setSessionCallback(sHandlerExecutor, callback).setId(TAG).build();
1192fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final MediaController2 controller = createController(mSession.getToken());
1193fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1194fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        callback.resetLatchCount(1);
1195fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        controller.subscribeRoutesInfo();
1196fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        assertTrue(callback.mLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1197fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1198fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        callback.resetLatchCount(1);
1199fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        controller.unsubscribeRoutesInfo();
1200fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        assertTrue(callback.mLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1201fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    }
1202fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1203fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    @Test
1204fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    public void testSelectRouteInfo() throws InterruptedException {
1205fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        prepareLooper();
1206fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final Bundle testRoute = new Bundle();
1207fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        testRoute.putString("id", "testRoute");
1208fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final TestSessionCallback callback = new TestSessionCallback() {
1209fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            @Override
1210fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            public void onSelectRoute(@NonNull MediaSession2 session,
1211fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                    @NonNull ControllerInfo controller, @NonNull Bundle route) {
1212fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                assertEquals(mContext.getPackageName(), controller.getPackageName());
1213fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                assertTrue(TestUtils.equals(route, testRoute));
1214fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                mLatch.countDown();
1215fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            }
1216fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        };
1217fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        mSession.close();
1218fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        mSession = new MediaSession2.Builder(mContext).setPlayer(mPlayer)
1219fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim                .setSessionCallback(sHandlerExecutor, callback).setId(TAG).build();
1220fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        final MediaController2 controller = createController(mSession.getToken());
1221fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1222fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        callback.resetLatchCount(1);
1223fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        controller.selectRoute(testRoute);
1224fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        assertTrue(callback.mLatch.await(TIMEOUT_MS, TimeUnit.MILLISECONDS));
1225fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    }
1226fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1227fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    @Test
1228fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_beforeConnected() throws InterruptedException {
1229fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1230fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        MediaController2 controller =
1231fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                createController(mSession.getToken(), false, null);
1232fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        controller.close();
1233fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1234fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1235fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1236fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_twice() {
1237fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1238fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
1239fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
1240fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1241fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1242fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1243fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_session() throws InterruptedException {
1244fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1245fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final String id = mSession.getToken().getId();
1246fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
1247fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // close is done immediately for session.
1248fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testNoInteraction();
1249fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1250fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Test whether the controller is notified about later close of the session or
1251fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // re-creation.
1252fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testControllerAfterSessionIsClosed(id);
1253fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1254fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1255fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1256fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_sessionService() throws InterruptedException {
1257fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1258fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testCloseFromService(MockMediaSessionService2.ID);
1259fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1260fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1261fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    @Test
1262fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    public void testClose_libraryService() throws InterruptedException {
1263fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        prepareLooper();
1264fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testCloseFromService(MockMediaLibraryService2.ID);
1265fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1266fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1267fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    private void testCloseFromService(String id) throws InterruptedException {
1268fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
1269fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestServiceRegistry.getInstance().setSessionServiceCallback(new SessionServiceCallback() {
1270fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1271fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onCreated() {
1272fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                // Do nothing.
1273fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1274fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1275fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1276fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void onDestroyed() {
1277fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
1278fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1279fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
1280fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController = createController(TestUtils.getServiceToken(mContext, id));
1281fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mController.close();
1282fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Wait until close triggers onDestroy() of the session service.
1283fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertTrue(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
1284fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertNull(TestServiceRegistry.getInstance().getServiceInstance());
1285fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testNoInteraction();
1286fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1287fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Test whether the controller is notified about later close of the session or
1288fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // re-creation.
1289fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testControllerAfterSessionIsClosed(id);
1290fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1291fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1292fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    private void testControllerAfterSessionIsClosed(final String id) throws InterruptedException {
1293fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // This cause session service to be died.
1294fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.close();
1295fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        waitForDisconnect(mController, true);
1296fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testNoInteraction();
1297fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1298fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Ensure that the controller cannot use newly create session with the same ID.
1299fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Recreated session has different session stub, so previously created controller
1300fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // shouldn't be available.
1301fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession = new MediaSession2.Builder(mContext)
1302fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setPlayer(mPlayer)
1303fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setSessionCallback(sHandlerExecutor, new SessionCallback() {})
1304fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                .setId(id).build();
1305fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        testNoInteraction();
1306fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1307fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1308fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // Test that mSession and mController doesn't interact.
1309fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // Note that this method can be called after the mSession is died, so mSession may not have
1310fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // valid player.
1311fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    private void testNoInteraction() throws InterruptedException {
1312fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // TODO: check that calls from the controller to session shouldn't be delivered.
1313fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1314fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        // Calls from the session to controller shouldn't be delivered.
1315fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch latch = new CountDownLatch(1);
1316fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        setRunnableForOnCustomCommand(mController, new Runnable() {
1317fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            @Override
1318fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            public void run() {
1319fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang                latch.countDown();
1320fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            }
1321fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        });
1322fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        SessionCommand2 customCommand = new SessionCommand2("testNoInteraction", null);
1323fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        mSession.sendCustomCommand(customCommand, null);
1324fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        assertFalse(latch.await(WAIT_TIME_MS, TimeUnit.MILLISECONDS));
1325fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        setRunnableForOnCustomCommand(mController, null);
1326fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1327fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1328fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    // TODO(jaewan): Add  test for service connect rejection, when we differentiate session
1329fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    //               active/inactive and connection accept/refuse
1330fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1331fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    class TestVolumeProvider extends VolumeProviderCompat {
1332fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        final CountDownLatch mLatch = new CountDownLatch(1);
1333fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        boolean mSetVolumeToCalled;
1334fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        boolean mAdjustVolumeCalled;
1335fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        int mVolume;
1336fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        int mDirection;
1337fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1338fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        TestVolumeProvider(int controlType, int maxVolume, int currentVolume) {
1339fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            super(controlType, maxVolume, currentVolume);
1340fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
1341fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1342fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        @Override
1343fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        public void onSetVolumeTo(int volume) {
1344fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mSetVolumeToCalled = true;
1345fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mVolume = volume;
1346fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mLatch.countDown();
1347fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
1348fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang
1349fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        @Override
1350fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        public void onAdjustVolume(int direction) {
1351fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mAdjustVolumeCalled = true;
1352fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mDirection = direction;
1353fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang            mLatch.countDown();
1354fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang        }
1355fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang    }
1356fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1357fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    class TestSessionCallback extends SessionCallback {
1358fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        CountDownLatch mLatch;
1359fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim
1360fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        void resetLatchCount(int count) {
1361fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim            mLatch = new CountDownLatch(count);
1362fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim        }
1363fa1b27e361831b01f2d97d816f48c39ccf0ff539Sungsoo Lim    }
1364fbbf807584a0fbe7a01a0aa9920330cad45689aaInsun Kang}
1365