blob: 3787245ec1645e61a2495e841448f2d5e17523f6 [file] [log] [blame]
// Copyright 2013 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 "chrome/browser/media/webrtc/webrtc_browsertest_common.h"
#include "base/files/file_util.h"
#include "base/path_service.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/test/test_timeouts.h"
#include "base/time/time.h"
#include "build/build_config.h"
#include "build/chromeos_buildflags.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ui/browser_tabstrip.h"
#include "chrome/common/chrome_paths.h"
#include "content/public/test/browser_test_utils.h"
namespace test {
// Relative to the chrome/test/data directory.
const base::FilePath::CharType kReferenceFilesDirName[] =
FILE_PATH_LITERAL("webrtc/resources");
const base::FilePath::CharType kReferenceFileName360p[] =
FILE_PATH_LITERAL("reference_video_640x360_30fps");
const base::FilePath::CharType kReferenceFileName720p[] =
FILE_PATH_LITERAL("reference_video_1280x720_30fps");
const base::FilePath::CharType kYuvFileExtension[] = FILE_PATH_LITERAL("yuv");
const base::FilePath::CharType kY4mFileExtension[] = FILE_PATH_LITERAL("y4m");
// This message describes how to modify your .gclient to get the reference
// video files downloaded for you.
const char kAdviseOnGclientSolution[] =
"To run this test, you must run download_from_google_storage --config\n"
"and follow the instructions (use 'browser' for project id)\n"
"You also need to add this solution to your .gclient:\n"
"{\n"
" \"name\" : \"webrtc.DEPS\",\n"
" \"url\" : \"https://chromium.googlesource.com/chromium/deps/"
"webrtc/webrtc.DEPS\",\n"
"}\n"
"and run gclient sync. This will download the required ref files.";
#if defined(THREAD_SANITIZER) || defined(MEMORY_SANITIZER) || \
defined(ADDRESS_SANITIZER)
#if BUILDFLAG(IS_CHROMEOS_ASH)
const int kDefaultPollIntervalMsec = 2000;
#else
const int kDefaultPollIntervalMsec = 1000;
#endif // BUILDFLAG(IS_CHROMEOS_ASH)
#else
#if BUILDFLAG(IS_CHROMEOS_ASH)
const int kDefaultPollIntervalMsec = 500;
#else
const int kDefaultPollIntervalMsec = 250;
#endif // BUILDFLAG(IS_CHROMEOS_ASH)
#endif
bool IsErrorResult(const std::string& result) {
return base::StartsWith(result, "failed-",
base::CompareCase::INSENSITIVE_ASCII);
}
base::FilePath GetReferenceFilesDir() {
base::FilePath test_data_dir;
base::PathService::Get(chrome::DIR_TEST_DATA, &test_data_dir);
return test_data_dir.Append(kReferenceFilesDirName);
}
base::FilePath GetToolForPlatform(const std::string& tool_name) {
base::FilePath tools_dir =
GetReferenceFilesDir().Append(FILE_PATH_LITERAL("tools"));
#if BUILDFLAG(IS_WIN)
return tools_dir
.Append(FILE_PATH_LITERAL("win"))
.AppendASCII(tool_name)
.AddExtension(FILE_PATH_LITERAL("exe"));
#elif BUILDFLAG(IS_MAC)
return tools_dir.Append(FILE_PATH_LITERAL("mac")).AppendASCII(tool_name);
#elif BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
return tools_dir.Append(FILE_PATH_LITERAL("linux")).AppendASCII(tool_name);
#else
CHECK(false) << "Can't retrieve tool " << tool_name << " on this platform.";
return base::FilePath();
#endif
}
bool HasReferenceFilesInCheckout() {
if (!base::PathExists(GetReferenceFilesDir())) {
LOG(ERROR)
<< "Cannot find the working directory for the reference video "
<< "files, expected at " << GetReferenceFilesDir().value() << ". " <<
kAdviseOnGclientSolution;
return false;
}
return HasYuvAndY4mFile(test::kReferenceFileName360p) &&
HasYuvAndY4mFile(test::kReferenceFileName720p);
}
bool HasYuvAndY4mFile(const base::FilePath::CharType* reference_file) {
base::FilePath webrtc_reference_video_yuv = GetReferenceFilesDir()
.Append(reference_file).AddExtension(kYuvFileExtension);
if (!base::PathExists(webrtc_reference_video_yuv)) {
LOG(ERROR)
<< "Missing YUV reference video to be used for quality"
<< " comparison, expected at " << webrtc_reference_video_yuv.value()
<< ". " << kAdviseOnGclientSolution;
return false;
}
base::FilePath webrtc_reference_video_y4m = GetReferenceFilesDir()
.Append(reference_file).AddExtension(kY4mFileExtension);
if (!base::PathExists(webrtc_reference_video_y4m)) {
LOG(ERROR)
<< "Missing Y4M reference video to be used for quality"
<< " comparison, expected at "<< webrtc_reference_video_y4m.value()
<< ". " << kAdviseOnGclientSolution;
return false;
}
return true;
}
bool SleepInJavascript(content::WebContents* tab_contents, int timeout_msec) {
const std::string javascript = base::StringPrintf(
"setTimeout(function() {"
" window.domAutomationController.send('sleep-ok');"
"}, %d)", timeout_msec);
std::string result;
bool ok = content::ExecuteScriptAndExtractString(
tab_contents, javascript, &result);
return ok && result == "sleep-ok";
}
bool PollingWaitUntil(const std::string& javascript,
const std::string& evaluates_to,
content::WebContents* tab_contents) {
return PollingWaitUntil(javascript, evaluates_to, tab_contents,
kDefaultPollIntervalMsec);
}
bool PollingWaitUntil(const std::string& javascript,
const std::string& evaluates_to,
content::WebContents* tab_contents,
int poll_interval_msec) {
base::Time start_time = base::Time::Now();
base::TimeDelta timeout = TestTimeouts::action_max_timeout();
std::string result;
while (base::Time::Now() - start_time < timeout) {
if (!content::ExecuteScriptAndExtractString(tab_contents, javascript,
&result)) {
LOG(ERROR) << "Failed to execute javascript " << javascript;
return false;
}
if (evaluates_to == result) {
return true;
} else if (IsErrorResult(result)) {
LOG(ERROR) << "|" << javascript << "| returned an error: " << result;
return false;
}
// Sleep a bit here to keep this loop from spinlocking too badly.
if (!SleepInJavascript(tab_contents, poll_interval_msec)) {
// TODO(phoglund): Figure out why this fails every now and then.
// It's not a huge deal if it does though.
LOG(ERROR) << "Failed to sleep.";
}
}
LOG(ERROR)
<< "Timed out while waiting for " << javascript
<< " to evaluate to " << evaluates_to << "; last result was '" << result
<< "'";
return false;
}
} // namespace test