// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/command_line.h" #include "base/debug/trace_event_impl.h" #include "base/json/json_reader.h" #include "base/strings/stringprintf.h" #include "base/strings/utf_string_conversions.h" #include "base/values.h" #include "content/browser/media/webrtc_internals.h" #include "content/browser/web_contents/web_contents_impl.h" #include "content/public/common/content_switches.h" #include "content/public/test/browser_test_utils.h" #include "content/shell/browser/shell.h" #include "content/test/content_browser_test.h" #include "content/test/content_browser_test_utils.h" #include "media/audio/audio_manager.h" #include "net/test/embedded_test_server/embedded_test_server.h" #include "testing/perf/perf_test.h" #if defined(OS_WIN) #include "base/win/windows_version.h" #endif const char kForceIsac16K[] = #ifdef OS_ANDROID // The default audio codec, Opus, doesn't work on Android. "true"; #else "false"; #endif namespace { static const char kGetUserMediaAndStop[] = "getUserMediaAndStop"; static const char kGetUserMediaAndWaitAndStop[] = "getUserMediaAndWaitAndStop"; static const char kGetUserMediaAndAnalyseAndStop[] = "getUserMediaAndAnalyseAndStop"; // Results returned by JS. static const char kOK[] = "OK"; static const char kGetUserMediaFailed[] = "GetUserMedia call failed with code undefined"; std::string GenerateGetUserMediaCall(const char* function_name, int min_width, int max_width, int min_height, int max_height, int min_frame_rate, int max_frame_rate) { return base::StringPrintf( "%s({video: {mandatory: {minWidth: %d, maxWidth: %d, " "minHeight: %d, maxHeight: %d, minFrameRate: %d, maxFrameRate: %d}, " "optional: []}});", function_name, min_width, max_width, min_height, max_height, min_frame_rate, max_frame_rate); } std::string GenerateGetUserMediaWithMandatorySourceID( const std::string& function_name, const std::string& audio_source_id, const std::string& video_source_id) { const std::string audio_constraint = "audio: {mandatory: { sourceId:\"" + audio_source_id + "\"}}, "; const std::string video_constraint = "video: {mandatory: { sourceId:\"" + video_source_id + "\"}}"; return function_name + "({" + audio_constraint + video_constraint + "});"; } std::string GenerateGetUserMediaWithOptionalSourceID( const std::string& function_name, const std::string& audio_source_id, const std::string& video_source_id) { const std::string audio_constraint = "audio: {optional: [{sourceId:\"" + audio_source_id + "\"}]}, "; const std::string video_constraint = "video: {optional: [{ sourceId:\"" + video_source_id + "\"}]}"; return function_name + "({" + audio_constraint + video_constraint + "});"; } } namespace content { class WebrtcBrowserTest: public ContentBrowserTest { public: WebrtcBrowserTest() {} virtual ~WebrtcBrowserTest() {} virtual void SetUpCommandLine(CommandLine* command_line) OVERRIDE { // We need fake devices in this test since we want to run on naked VMs. We // assume these switches are set by default in content_browsertests. ASSERT_TRUE(CommandLine::ForCurrentProcess()->HasSwitch( switches::kUseFakeDeviceForMediaStream)); ASSERT_TRUE(CommandLine::ForCurrentProcess()->HasSwitch( switches::kUseFakeUIForMediaStream)); // The video playback will not work without a GPU, so force its use here. // This may not be available on all VMs though. command_line->AppendSwitch(switches::kUseGpuInTests); } void DumpChromeTraceCallback( const scoped_refptr& events, bool has_more_events) { // Convert the dump output into a correct JSON List. std::string contents = "[" + events->data() + "]"; int error_code; std::string error_message; scoped_ptr value( base::JSONReader::ReadAndReturnError(contents, base::JSON_ALLOW_TRAILING_COMMAS, &error_code, &error_message)); ASSERT_TRUE(value.get() != NULL) << error_message; EXPECT_EQ(value->GetType(), base::Value::TYPE_LIST); base::ListValue* values; ASSERT_TRUE(value->GetAsList(&values)); int duration_ns = 0; std::string samples_duration; double timestamp_ns = 0.0; double previous_timestamp_ns = 0.0; std::string samples_interarrival_ns; for (base::ListValue::iterator it = values->begin(); it != values->end(); ++it) { const base::DictionaryValue* dict; EXPECT_TRUE((*it)->GetAsDictionary(&dict)); if (dict->GetInteger("dur", &duration_ns)) samples_duration.append(base::StringPrintf("%d,", duration_ns)); if (dict->GetDouble("ts", ×tamp_ns)) { if (previous_timestamp_ns) { samples_interarrival_ns.append( base::StringPrintf("%f,", timestamp_ns - previous_timestamp_ns)); } previous_timestamp_ns = timestamp_ns; } } ASSERT_GT(samples_duration.size(), 0u) << "Could not collect any samples during test, this is bad"; perf_test::PrintResultList("video_capture", "", "sample_duration", samples_duration, "ns", true); perf_test::PrintResultList("video_capture", "", "interarrival_time", samples_interarrival_ns, "ns", true); } void GetSources(std::vector* audio_ids, std::vector* video_ids) { GURL url(embedded_test_server()->GetURL("/media/getusermedia.html")); NavigateToURL(shell(), url); std::string sources_as_json = ExecuteJavascriptAndReturnResult( "getSources()"); EXPECT_FALSE(sources_as_json.empty()); int error_code; std::string error_message; scoped_ptr value( base::JSONReader::ReadAndReturnError(sources_as_json, base::JSON_ALLOW_TRAILING_COMMAS, &error_code, &error_message)); ASSERT_TRUE(value.get() != NULL) << error_message; EXPECT_EQ(value->GetType(), base::Value::TYPE_LIST); base::ListValue* values; ASSERT_TRUE(value->GetAsList(&values)); for (base::ListValue::iterator it = values->begin(); it != values->end(); ++it) { const base::DictionaryValue* dict; std::string kind; std::string id; ASSERT_TRUE((*it)->GetAsDictionary(&dict)); ASSERT_TRUE(dict->GetString("kind", &kind)); ASSERT_TRUE(dict->GetString("id", &id)); ASSERT_FALSE(id.empty()); EXPECT_TRUE(kind == "audio" || kind == "video"); if (kind == "audio") { audio_ids->push_back(id); } else if (kind == "video") { video_ids->push_back(id); } } ASSERT_FALSE(audio_ids->empty()); ASSERT_FALSE(video_ids->empty()); } protected: bool ExecuteJavascript(const std::string& javascript) { return ExecuteScript(shell()->web_contents(), javascript); } // Executes |javascript|. The script is required to use // window.domAutomationController.send to send a string value back to here. std::string ExecuteJavascriptAndReturnResult(const std::string& javascript) { std::string result; EXPECT_TRUE(ExecuteScriptAndExtractString(shell()->web_contents(), javascript, &result)); return result; } void ExpectTitle(const std::string& expected_title) const { base::string16 expected_title16(ASCIIToUTF16(expected_title)); TitleWatcher title_watcher(shell()->web_contents(), expected_title16); EXPECT_EQ(expected_title16, title_watcher.WaitAndGetTitle()); } }; // These tests will all make a getUserMedia call with different constraints and // see that the success callback is called. If the error callback is called or // none of the callbacks are called the tests will simply time out and fail. IN_PROC_BROWSER_TEST_F(WebrtcBrowserTest, GetVideoStreamAndStop) { ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); GURL url(embedded_test_server()->GetURL("/media/getusermedia.html")); NavigateToURL(shell(), url); EXPECT_TRUE(ExecuteJavascript( base::StringPrintf("%s({video: true});", kGetUserMediaAndStop))); ExpectTitle("OK"); } IN_PROC_BROWSER_TEST_F(WebrtcBrowserTest, GetAudioAndVideoStreamAndStop) { ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); GURL url(embedded_test_server()->GetURL("/media/getusermedia.html")); NavigateToURL(shell(), url); EXPECT_TRUE(ExecuteJavascript(base::StringPrintf( "%s({video: true, audio: true});", kGetUserMediaAndStop))); ExpectTitle("OK"); } IN_PROC_BROWSER_TEST_F(WebrtcBrowserTest, GetAudioAndVideoStreamAndClone) { ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); GURL url(embedded_test_server()->GetURL("/media/getusermedia.html")); NavigateToURL(shell(), url); EXPECT_TRUE(ExecuteJavascript("getUserMediaAndClone();")); ExpectTitle("OK"); } IN_PROC_BROWSER_TEST_F(WebrtcBrowserTest, GetUserMediaWithMandatorySourceID) { ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); std::vector audio_ids; std::vector video_ids; GetSources(&audio_ids, &video_ids); GURL url(embedded_test_server()->GetURL("/media/getusermedia.html")); // Test all combinations of mandatory sourceID; for (std::vector::const_iterator video_it = video_ids.begin(); video_it != video_ids.end(); ++video_it) { for (std::vector::const_iterator audio_it = audio_ids.begin(); audio_it != audio_ids.end(); ++audio_it) { NavigateToURL(shell(), url); EXPECT_EQ(kOK, ExecuteJavascriptAndReturnResult( GenerateGetUserMediaWithMandatorySourceID( kGetUserMediaAndStop, *audio_it, *video_it))); } } } IN_PROC_BROWSER_TEST_F(WebrtcBrowserTest, GetUserMediaWithInvalidMandatorySourceID) { ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); std::vector audio_ids; std::vector video_ids; GetSources(&audio_ids, &video_ids); GURL url(embedded_test_server()->GetURL("/media/getusermedia.html")); // Test with invalid mandatory audio sourceID. NavigateToURL(shell(), url); EXPECT_EQ(kGetUserMediaFailed, ExecuteJavascriptAndReturnResult( GenerateGetUserMediaWithMandatorySourceID( kGetUserMediaAndStop, "something invalid", video_ids[0]))); // Test with invalid mandatory video sourceID. EXPECT_EQ(kGetUserMediaFailed, ExecuteJavascriptAndReturnResult( GenerateGetUserMediaWithMandatorySourceID( kGetUserMediaAndStop, audio_ids[0], "something invalid"))); // Test with empty mandatory audio sourceID. EXPECT_EQ(kGetUserMediaFailed, ExecuteJavascriptAndReturnResult( GenerateGetUserMediaWithMandatorySourceID( kGetUserMediaAndStop, "", video_ids[0]))); } IN_PROC_BROWSER_TEST_F(WebrtcBrowserTest, GetUserMediaWithOptionalSourceID) { ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); std::vector audio_ids; std::vector video_ids; GetSources(&audio_ids, &video_ids); GURL url(embedded_test_server()->GetURL("/media/getusermedia.html")); NavigateToURL(shell(), url); // Test all combinations of mandatory sourceID; for (std::vector::const_iterator video_it = video_ids.begin(); video_it != video_ids.end(); ++video_it) { for (std::vector::const_iterator audio_it = audio_ids.begin(); audio_it != audio_ids.end(); ++audio_it) { EXPECT_EQ(kOK, ExecuteJavascriptAndReturnResult( GenerateGetUserMediaWithOptionalSourceID( kGetUserMediaAndStop, *audio_it, *video_it))); } } } IN_PROC_BROWSER_TEST_F(WebrtcBrowserTest, GetUserMediaWithInvalidOptionalSourceID) { ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); std::vector audio_ids; std::vector video_ids; GetSources(&audio_ids, &video_ids); GURL url(embedded_test_server()->GetURL("/media/getusermedia.html")); // Test with invalid optional audio sourceID. NavigateToURL(shell(), url); EXPECT_EQ(kOK, ExecuteJavascriptAndReturnResult( GenerateGetUserMediaWithOptionalSourceID( kGetUserMediaAndStop, "something invalid", video_ids[0]))); // Test with invalid optional video sourceID. EXPECT_EQ(kOK, ExecuteJavascriptAndReturnResult( GenerateGetUserMediaWithOptionalSourceID( kGetUserMediaAndStop, audio_ids[0], "something invalid"))); // Test with empty optional audio sourceID. EXPECT_EQ(kOK, ExecuteJavascriptAndReturnResult( GenerateGetUserMediaWithOptionalSourceID( kGetUserMediaAndStop, "", video_ids[0]))); } #if defined(OS_LINUX) && !defined(OS_CHROMEOS) && defined(ARCH_CPU_ARM_FAMILY) // Timing out on ARM linux bot: http://crbug.com/238490 #define MAYBE_CanSetupVideoCall DISABLED_CanSetupVideoCall #else #define MAYBE_CanSetupVideoCall CanSetupVideoCall #endif // These tests will make a complete PeerConnection-based call and verify that // video is playing for the call. IN_PROC_BROWSER_TEST_F(WebrtcBrowserTest, MAYBE_CanSetupVideoCall) { ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); GURL url(embedded_test_server()->GetURL("/media/peerconnection-call.html")); NavigateToURL(shell(), url); EXPECT_TRUE(ExecuteJavascript("call({video: true});")); ExpectTitle("OK"); } // This test will make a simple getUserMedia page, verify that video is playing // in a simple local