1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/command_line.h"
6 #include "base/json/json_reader.h"
7 #include "base/strings/string_util.h"
8 #include "chrome/browser/media/webrtc_browsertest_base.h"
9 #include "chrome/browser/media/webrtc_browsertest_common.h"
10 #include "chrome/browser/ui/browser.h"
11 #include "chrome/browser/ui/browser_tabstrip.h"
12 #include "chrome/browser/ui/tabs/tab_strip_model.h"
13 #include "chrome/common/chrome_switches.h"
14 #include "chrome/test/base/in_process_browser_test.h"
15 #include "chrome/test/base/ui_test_utils.h"
16 #include "content/public/test/browser_test_utils.h"
17 #include "media/audio/audio_manager.h"
18 #include "media/base/media_switches.h"
19 #include "net/test/embedded_test_server/embedded_test_server.h"
23 const char kMainWebrtcTestHtmlPage
[] = "/webrtc/webrtc_jsep01_test.html";
25 const char kDeviceKindAudioInput
[] = "audioinput";
26 const char kDeviceKindVideoInput
[] = "videoinput";
27 const char kDeviceKindAudioOutput
[] = "audiooutput";
29 const char kSourceKindAudioInput
[] = "audio";
30 const char kSourceKindVideoInput
[] = "video";
34 // Integration test for WebRTC getMediaDevices. It always uses fake devices.
35 // It needs to be a browser test (and not content browser test) to be able to
36 // test that labels are cleared or not depending on if access to devices has
38 class WebRtcGetMediaDevicesBrowserTest
39 : public WebRtcTestBase
,
40 public testing::WithParamInterface
<bool> {
42 WebRtcGetMediaDevicesBrowserTest()
43 : has_audio_output_devices_initialized_(false),
44 has_audio_output_devices_(false) {}
46 virtual void SetUpInProcessBrowserTestFixture() OVERRIDE
{
47 DetectErrorsInJavaScript(); // Look for errors in our rather complex js.
50 virtual void SetUpCommandLine(CommandLine
* command_line
) OVERRIDE
{
51 // Ensure the infobar is enabled, since we expect that in this test.
52 EXPECT_FALSE(command_line
->HasSwitch(switches::kUseFakeUIForMediaStream
));
54 // Always use fake devices.
55 command_line
->AppendSwitch(switches::kUseFakeDeviceForMediaStream
);
59 // This is used for media devices and sources.
60 struct MediaDeviceInfo
{
61 std::string device_id
; // Domain specific device ID.
67 bool HasOutputDevices() {
68 // There's no fake audio output devices supported yet. We can't test audio
69 // output devices on bots with no output devices, so skip testing for that
70 // on such bots. We cache the result since querying for devices can take
72 if (!has_audio_output_devices_initialized_
) {
73 has_audio_output_devices_
=
74 media::AudioManager::Get()->HasAudioOutputDevices();
75 has_audio_output_devices_initialized_
= true;
77 return has_audio_output_devices_
;
80 // If |get_sources| is true, use getSources API and leave groupId empty,
81 // otherwise use getMediaDevices API.
82 void GetMediaDevicesOrSources(content::WebContents
* tab
,
83 std::vector
<MediaDeviceInfo
>* devices
,
85 std::string devices_as_json
=
86 ExecuteJavascript(get_sources
? "getSources()" : "getMediaDevices()",
88 EXPECT_FALSE(devices_as_json
.empty());
91 std::string error_message
;
92 scoped_ptr
<base::Value
> value(
93 base::JSONReader::ReadAndReturnError(devices_as_json
,
94 base::JSON_ALLOW_TRAILING_COMMAS
,
98 ASSERT_TRUE(value
.get() != NULL
) << error_message
;
99 EXPECT_EQ(value
->GetType(), base::Value::TYPE_LIST
);
101 base::ListValue
* values
;
102 ASSERT_TRUE(value
->GetAsList(&values
));
103 ASSERT_FALSE(values
->empty());
104 bool found_audio_input
= false;
105 bool found_video_input
= false;
106 bool found_audio_output
= false;
108 for (base::ListValue::iterator it
= values
->begin();
109 it
!= values
->end(); ++it
) {
110 const base::DictionaryValue
* dict
;
111 MediaDeviceInfo device
;
112 ASSERT_TRUE((*it
)->GetAsDictionary(&dict
));
113 ASSERT_TRUE(dict
->GetString(get_sources
? "id" : "deviceId",
115 ASSERT_TRUE(dict
->GetString("kind", &device
.kind
));
116 ASSERT_TRUE(dict
->GetString("label", &device
.label
));
118 ASSERT_TRUE(dict
->GetString("groupId", &device
.group_id
));
120 // Should be HMAC SHA256.
121 EXPECT_EQ(64ul, device
.device_id
.length());
122 EXPECT_TRUE(base::ContainsOnlyChars(device
.device_id
,
123 "0123456789abcdef"));
125 const char* kAudioInputKind
=
126 get_sources
? kSourceKindAudioInput
: kDeviceKindAudioInput
;
127 const char* kVideoInputKind
=
128 get_sources
? kSourceKindVideoInput
: kDeviceKindVideoInput
;
130 EXPECT_TRUE(device
.kind
== kAudioInputKind
||
131 device
.kind
== kVideoInputKind
);
133 EXPECT_TRUE(device
.kind
== kAudioInputKind
||
134 device
.kind
== kVideoInputKind
||
135 device
.kind
== kDeviceKindAudioOutput
);
137 if (device
.kind
== kAudioInputKind
) {
138 found_audio_input
= true;
139 } else if (device
.kind
== kVideoInputKind
) {
140 found_video_input
= true;
142 found_audio_output
= true;
145 // getSources doesn't have group ID support. getMediaDevices doesn't have
146 // group ID support for video input devices.
147 if (get_sources
|| device
.kind
== kDeviceKindVideoInput
) {
148 EXPECT_TRUE(device
.group_id
.empty());
150 EXPECT_FALSE(device
.group_id
.empty());
153 devices
->push_back(device
);
156 EXPECT_TRUE(found_audio_input
);
157 EXPECT_TRUE(found_video_input
);
159 EXPECT_FALSE(found_audio_output
);
161 EXPECT_EQ(HasOutputDevices(), found_audio_output
);
165 void GetMediaDevices(content::WebContents
* tab
,
166 std::vector
<MediaDeviceInfo
>* devices
) {
167 GetMediaDevicesOrSources(tab
, devices
, false);
170 void GetSources(content::WebContents
* tab
,
171 std::vector
<MediaDeviceInfo
>* sources
) {
172 GetMediaDevicesOrSources(tab
, sources
, true);
175 bool has_audio_output_devices_initialized_
;
176 bool has_audio_output_devices_
;
179 static const bool kParamsToRunTestsWith
[] = { false, true };
180 INSTANTIATE_TEST_CASE_P(WebRtcGetMediaDevicesBrowserTests
,
181 WebRtcGetMediaDevicesBrowserTest
,
182 testing::ValuesIn(kParamsToRunTestsWith
));
184 // getMediaDevices has been removed and will be replaced
185 // MediaDevices.enumerateDevices. http://crbug.com/388648.
186 IN_PROC_BROWSER_TEST_P(WebRtcGetMediaDevicesBrowserTest
,
187 DISABLED_GetMediaDevicesWithoutAccess
) {
188 ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady());
189 GURL
url(embedded_test_server()->GetURL(kMainWebrtcTestHtmlPage
));
190 ui_test_utils::NavigateToURL(browser(), url
);
191 content::WebContents
* tab
=
192 browser()->tab_strip_model()->GetActiveWebContents();
194 std::vector
<MediaDeviceInfo
> devices
;
195 GetMediaDevices(tab
, &devices
);
197 // Labels should be empty if access has not been allowed.
198 for (std::vector
<MediaDeviceInfo
>::iterator it
= devices
.begin();
199 it
!= devices
.end(); ++it
) {
200 EXPECT_TRUE(it
->label
.empty());
204 // getMediaDevices has been removed and will be replaced
205 // MediaDevices.enumerateDevices. http://crbug.com/388648.
206 // Disabled, fails due to http://crbug.com/382391.
207 IN_PROC_BROWSER_TEST_P(WebRtcGetMediaDevicesBrowserTest
,
208 DISABLED_GetMediaDevicesWithAccess
) {
209 ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady());
210 GURL
url(embedded_test_server()->GetURL(kMainWebrtcTestHtmlPage
));
211 ui_test_utils::NavigateToURL(browser(), url
);
212 content::WebContents
* tab
=
213 browser()->tab_strip_model()->GetActiveWebContents();
215 GetUserMediaAndAccept(tab
);
217 std::vector
<MediaDeviceInfo
> devices
;
218 GetMediaDevices(tab
, &devices
);
220 // Labels should be non-empty if access has been allowed.
221 for (std::vector
<MediaDeviceInfo
>::iterator it
= devices
.begin();
222 it
!= devices
.end(); ++it
) {
223 EXPECT_TRUE(!it
->label
.empty());
227 // getMediaDevices has been removed and will be replaced
228 // MediaDevices.enumerateDevices. http://crbug.com/388648.
229 IN_PROC_BROWSER_TEST_P(WebRtcGetMediaDevicesBrowserTest
,
230 DISABLED_GetMediaDevicesEqualsGetSourcesWithoutAccess
) {
231 ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady());
232 GURL
url(embedded_test_server()->GetURL(kMainWebrtcTestHtmlPage
));
233 ui_test_utils::NavigateToURL(browser(), url
);
234 content::WebContents
* tab
=
235 browser()->tab_strip_model()->GetActiveWebContents();
237 std::vector
<MediaDeviceInfo
> devices
;
238 GetMediaDevices(tab
, &devices
);
240 std::vector
<MediaDeviceInfo
> sources
;
241 GetSources(tab
, &sources
);
243 std::vector
<MediaDeviceInfo
>::iterator sources_it
= sources
.begin();
244 for (std::vector
<MediaDeviceInfo
>::iterator devices_it
= devices
.begin();
245 devices_it
!= devices
.end(); ++devices_it
) {
246 if (devices_it
->kind
== kDeviceKindAudioOutput
)
248 EXPECT_STREQ(devices_it
->device_id
.c_str(), sources_it
->device_id
.c_str());
249 if (devices_it
->kind
== kDeviceKindAudioInput
) {
250 EXPECT_STREQ(kSourceKindAudioInput
, sources_it
->kind
.c_str());
252 EXPECT_STREQ(kSourceKindVideoInput
, sources_it
->kind
.c_str());
254 EXPECT_TRUE(devices_it
->label
.empty());
255 EXPECT_TRUE(sources_it
->label
.empty());
258 EXPECT_EQ(sources
.end(), sources_it
);
261 // getMediaDevices has been removed and will be replaced
262 // MediaDevices.enumerateDevices. http://crbug.com/388648.
263 // Disabled, fails due to http://crbug.com/382391.
264 IN_PROC_BROWSER_TEST_P(WebRtcGetMediaDevicesBrowserTest
,
265 DISABLED_GetMediaDevicesEqualsGetSourcesWithAccess
) {
266 ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady());
267 GURL
url(embedded_test_server()->GetURL(kMainWebrtcTestHtmlPage
));
268 ui_test_utils::NavigateToURL(browser(), url
);
269 content::WebContents
* tab
=
270 browser()->tab_strip_model()->GetActiveWebContents();
272 GetUserMediaAndAccept(tab
);
274 std::vector
<MediaDeviceInfo
> devices
;
275 GetMediaDevices(tab
, &devices
);
277 std::vector
<MediaDeviceInfo
> sources
;
278 GetSources(tab
, &sources
);
280 std::vector
<MediaDeviceInfo
>::iterator sources_it
= sources
.begin();
281 for (std::vector
<MediaDeviceInfo
>::iterator devices_it
= devices
.begin();
282 devices_it
!= devices
.end(); ++devices_it
) {
283 if (devices_it
->kind
== kDeviceKindAudioOutput
)
285 EXPECT_STREQ(devices_it
->device_id
.c_str(), sources_it
->device_id
.c_str());
286 if (devices_it
->kind
== kDeviceKindAudioInput
) {
287 EXPECT_STREQ(kSourceKindAudioInput
, sources_it
->kind
.c_str());
289 EXPECT_STREQ(kSourceKindVideoInput
, sources_it
->kind
.c_str());
291 EXPECT_TRUE(!devices_it
->label
.empty());
292 EXPECT_STREQ(devices_it
->label
.c_str(), sources_it
->label
.c_str());
295 EXPECT_EQ(sources
.end(), sources_it
);