• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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.
4 
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"
20 
21 namespace {
22 
23 const char kMainWebrtcTestHtmlPage[] = "/webrtc/webrtc_jsep01_test.html";
24 
25 const char kDeviceKindAudioInput[] = "audioinput";
26 const char kDeviceKindVideoInput[] = "videoinput";
27 const char kDeviceKindAudioOutput[] = "audiooutput";
28 
29 const char kSourceKindAudioInput[] = "audio";
30 const char kSourceKindVideoInput[] = "video";
31 
32 }  // namespace
33 
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
37 // been granted.
38 class WebRtcGetMediaDevicesBrowserTest
39     : public WebRtcTestBase,
40       public testing::WithParamInterface<bool> {
41  public:
WebRtcGetMediaDevicesBrowserTest()42   WebRtcGetMediaDevicesBrowserTest()
43       : has_audio_output_devices_initialized_(false),
44         has_audio_output_devices_(false) {}
45 
SetUpInProcessBrowserTestFixture()46   virtual void SetUpInProcessBrowserTestFixture() OVERRIDE {
47     DetectErrorsInJavaScript();  // Look for errors in our rather complex js.
48   }
49 
SetUpCommandLine(CommandLine * command_line)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));
53 
54     // Always use fake devices.
55     command_line->AppendSwitch(switches::kUseFakeDeviceForMediaStream);
56   }
57 
58  protected:
59   // This is used for media devices and sources.
60   struct MediaDeviceInfo {
61     std::string device_id;  // Domain specific device ID.
62     std::string kind;
63     std::string label;
64     std::string group_id;
65   };
66 
HasOutputDevices()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
71     // considerable time.
72     if (!has_audio_output_devices_initialized_) {
73       has_audio_output_devices_ =
74           media::AudioManager::Get()->HasAudioOutputDevices();
75       has_audio_output_devices_initialized_ = true;
76     }
77     return has_audio_output_devices_;
78   }
79 
80   // If |get_sources| is true, use getSources API and leave groupId empty,
81   // otherwise use getMediaDevices API.
GetMediaDevicesOrSources(content::WebContents * tab,std::vector<MediaDeviceInfo> * devices,bool get_sources)82   void GetMediaDevicesOrSources(content::WebContents* tab,
83                                 std::vector<MediaDeviceInfo>* devices,
84                                 bool get_sources) {
85     std::string devices_as_json =
86         ExecuteJavascript(get_sources ? "getSources()" : "getMediaDevices()",
87                           tab);
88     EXPECT_FALSE(devices_as_json.empty());
89 
90     int error_code;
91     std::string error_message;
92     scoped_ptr<base::Value> value(
93         base::JSONReader::ReadAndReturnError(devices_as_json,
94                                              base::JSON_ALLOW_TRAILING_COMMAS,
95                                              &error_code,
96                                              &error_message));
97 
98     ASSERT_TRUE(value.get() != NULL) << error_message;
99     EXPECT_EQ(value->GetType(), base::Value::TYPE_LIST);
100 
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;
107 
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",
114                                   &device.device_id));
115       ASSERT_TRUE(dict->GetString("kind", &device.kind));
116       ASSERT_TRUE(dict->GetString("label", &device.label));
117       if (!get_sources)
118         ASSERT_TRUE(dict->GetString("groupId", &device.group_id));
119 
120       // Should be HMAC SHA256.
121       EXPECT_EQ(64ul, device.device_id.length());
122       EXPECT_TRUE(base::ContainsOnlyChars(device.device_id,
123                                           "0123456789abcdef"));
124 
125       const char* kAudioInputKind =
126           get_sources ? kSourceKindAudioInput : kDeviceKindAudioInput;
127       const char* kVideoInputKind =
128           get_sources ? kSourceKindVideoInput : kDeviceKindVideoInput;
129       if (get_sources) {
130         EXPECT_TRUE(device.kind == kAudioInputKind ||
131                     device.kind == kVideoInputKind);
132       } else {
133         EXPECT_TRUE(device.kind == kAudioInputKind ||
134                     device.kind == kVideoInputKind ||
135                     device.kind == kDeviceKindAudioOutput);
136       }
137       if (device.kind == kAudioInputKind) {
138         found_audio_input = true;
139       } else if (device.kind == kVideoInputKind) {
140         found_video_input = true;
141       } else {
142         found_audio_output = true;
143       }
144 
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());
149       } else {
150         EXPECT_FALSE(device.group_id.empty());
151       }
152 
153       devices->push_back(device);
154     }
155 
156     EXPECT_TRUE(found_audio_input);
157     EXPECT_TRUE(found_video_input);
158     if (get_sources) {
159       EXPECT_FALSE(found_audio_output);
160     } else {
161       EXPECT_EQ(HasOutputDevices(), found_audio_output);
162     }
163   }
164 
GetMediaDevices(content::WebContents * tab,std::vector<MediaDeviceInfo> * devices)165   void GetMediaDevices(content::WebContents* tab,
166                      std::vector<MediaDeviceInfo>* devices) {
167     GetMediaDevicesOrSources(tab, devices, false);
168   }
169 
GetSources(content::WebContents * tab,std::vector<MediaDeviceInfo> * sources)170   void GetSources(content::WebContents* tab,
171                   std::vector<MediaDeviceInfo>* sources) {
172     GetMediaDevicesOrSources(tab, sources, true);
173   }
174 
175   bool has_audio_output_devices_initialized_;
176   bool has_audio_output_devices_;
177 };
178 
179 static const bool kParamsToRunTestsWith[] = { false, true };
180 INSTANTIATE_TEST_CASE_P(WebRtcGetMediaDevicesBrowserTests,
181                         WebRtcGetMediaDevicesBrowserTest,
182                         testing::ValuesIn(kParamsToRunTestsWith));
183 
184 // getMediaDevices has been removed and will be replaced
185 // MediaDevices.enumerateDevices. http://crbug.com/388648.
IN_PROC_BROWSER_TEST_P(WebRtcGetMediaDevicesBrowserTest,DISABLED_GetMediaDevicesWithoutAccess)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();
193 
194   std::vector<MediaDeviceInfo> devices;
195   GetMediaDevices(tab, &devices);
196 
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());
201   }
202 }
203 
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.
IN_PROC_BROWSER_TEST_P(WebRtcGetMediaDevicesBrowserTest,DISABLED_GetMediaDevicesWithAccess)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();
214 
215   GetUserMediaAndAccept(tab);
216 
217   std::vector<MediaDeviceInfo> devices;
218   GetMediaDevices(tab, &devices);
219 
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());
224   }
225 }
226 
227 // getMediaDevices has been removed and will be replaced
228 // MediaDevices.enumerateDevices. http://crbug.com/388648.
IN_PROC_BROWSER_TEST_P(WebRtcGetMediaDevicesBrowserTest,DISABLED_GetMediaDevicesEqualsGetSourcesWithoutAccess)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();
236 
237   std::vector<MediaDeviceInfo> devices;
238   GetMediaDevices(tab, &devices);
239 
240   std::vector<MediaDeviceInfo> sources;
241   GetSources(tab, &sources);
242 
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)
247       continue;
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());
251     } else {
252       EXPECT_STREQ(kSourceKindVideoInput, sources_it->kind.c_str());
253     }
254     EXPECT_TRUE(devices_it->label.empty());
255     EXPECT_TRUE(sources_it->label.empty());
256     ++sources_it;
257   }
258   EXPECT_EQ(sources.end(), sources_it);
259 }
260 
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.
IN_PROC_BROWSER_TEST_P(WebRtcGetMediaDevicesBrowserTest,DISABLED_GetMediaDevicesEqualsGetSourcesWithAccess)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();
271 
272   GetUserMediaAndAccept(tab);
273 
274   std::vector<MediaDeviceInfo> devices;
275   GetMediaDevices(tab, &devices);
276 
277   std::vector<MediaDeviceInfo> sources;
278   GetSources(tab, &sources);
279 
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)
284       continue;
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());
288     } else {
289       EXPECT_STREQ(kSourceKindVideoInput, sources_it->kind.c_str());
290     }
291     EXPECT_TRUE(!devices_it->label.empty());
292     EXPECT_STREQ(devices_it->label.c_str(), sources_it->label.c_str());
293     ++sources_it;
294   }
295   EXPECT_EQ(sources.end(), sources_it);
296 }
297