blob: 98817b8dfe5845bc9c2127fd6447e777f7303f1e [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 <stdint.h>
#include <memory>
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/callback.h"
#include "base/command_line.h"
#include "base/compiler_specific.h"
#include "base/location.h"
#include "base/macros.h"
#include "base/memory/ref_counted.h"
#include "base/run_loop.h"
#include "base/single_thread_task_runner.h"
#include "base/test/bind_test_util.h"
#include "base/test/scoped_feature_list.h"
#include "base/threading/thread_task_runner_handle.h"
#include "build/build_config.h"
#include "chrome/browser/local_discovery/test_service_discovery_client.h"
#include "chrome/browser/media/router/media_router_feature.h"
#include "chrome/browser/media/router/providers/cast/dual_media_sink_service.h"
#include "chrome/browser/media/router/test/noop_dual_media_sink_service.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/profiles/profile_manager.h"
#include "chrome/browser/signin/chrome_signin_client_factory.h"
#include "chrome/browser/signin/identity_manager_factory.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/webui/local_discovery/local_discovery_ui_handler.h"
#include "chrome/common/chrome_constants.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/common/url_constants.h"
#include "chrome/test/base/ui_test_utils.h"
#include "chrome/test/base/web_ui_browser_test.h"
#include "google_apis/gaia/gaia_urls.h"
#include "net/http/http_status_code.h"
#include "services/identity/public/cpp/identity_test_utils.h"
#include "services/network/public/cpp/weak_wrapper_shared_url_loader_factory.h"
#include "services/network/test/test_url_loader_factory.h"
#if defined(OS_CHROMEOS)
#include "chrome/browser/browser_process.h"
#include "chrome/browser/chromeos/printing/cups_printers_manager_factory.h"
#include "chrome/common/pref_names.h"
#include "chromeos/components/account_manager/account_manager.h"
#include "chromeos/components/account_manager/account_manager_factory.h"
#include "chromeos/constants/chromeos_switches.h"
#include "components/keyed_service/content/browser_context_keyed_service_factory.h"
#include "components/prefs/pref_service.h"
#endif
using testing::AnyNumber;
using testing::AtLeast;
using testing::DoAll;
using testing::DoDefault;
using testing::InSequence;
using testing::InvokeWithoutArgs;
using testing::Return;
using testing::StrictMock;
namespace local_discovery {
namespace {
const uint8_t kQueryData[] = {
// Header
0x00, 0x00,
0x00, 0x00, // Flags not set.
0x00, 0x01, // Set QDCOUNT (question count) to 1, all the
// rest are 0 for a query.
0x00, 0x00,
0x00, 0x00,
0x00, 0x00,
// Question
0x07, '_', 'p', 'r', 'i', 'v', 'e', 't',
0x04, '_', 't', 'c', 'p',
0x05, 'l', 'o', 'c', 'a', 'l',
0x00,
0x00, 0x0c, // QTYPE: A query.
0x00, 0x01, // QCLASS: IN class. Unicast bit not set.
};
const uint8_t kAnnouncePacket[] = {
// Header
0x00, 0x00, // ID is zeroed out
0x80, 0x00, // Standard query response, no error
0x00, 0x00, // No questions (for simplicity)
0x00, 0x05, // 5 RR (answers)
0x00, 0x00, // 0 authority RRs
0x00, 0x00, // 0 additional RRs
0x07, '_', 'p', 'r', 'i', 'v', 'e', 't',
0x04, '_', 't', 'c', 'p',
0x05, 'l', 'o', 'c', 'a', 'l',
0x00,
0x00, 0x0c, // TYPE is PTR.
0x00, 0x01, // CLASS is IN.
0x00, 0x00, // TTL (4 bytes) is 32768 second.
0x10, 0x00,
0x00, 0x0c, // RDLENGTH is 12 bytes.
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0xc0, 0x0c,
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0xc0, 0x0c,
0x00, 0x10, // TYPE is TXT.
0x00, 0x01, // CLASS is IN.
0x00, 0x00, // TTL (4 bytes) is 32768 seconds.
0x01, 0x00,
0x00, 0x41, // RDLENGTH is 69 bytes.
0x03, 'i', 'd', '=',
0x10, 't', 'y', '=', 'S', 'a', 'm', 'p', 'l', 'e', ' ',
'd', 'e', 'v', 'i', 'c', 'e',
0x1e, 'n', 'o', 't', 'e', '=',
'S', 'a', 'm', 'p', 'l', 'e', ' ', 'd', 'e', 'v', 'i', 'c', 'e', ' ',
'd', 'e', 's', 'c', 'r', 'i', 'p', 't', 'i', 'o', 'n',
0x0c, 't', 'y', 'p', 'e', '=', 'p', 'r', 'i', 'n', 't', 'e', 'r',
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0xc0, 0x0c,
0x00, 0x21, // Type is SRV
0x00, 0x01, // CLASS is IN
0x00, 0x00, // TTL (4 bytes) is 32768 second.
0x10, 0x00,
0x00, 0x17, // RDLENGTH is 23
0x00, 0x00,
0x00, 0x00,
0x22, 0xb8, // port 8888
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0x05, 'l', 'o', 'c', 'a', 'l',
0x00,
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0x05, 'l', 'o', 'c', 'a', 'l',
0x00,
0x00, 0x01, // Type is A
0x00, 0x01, // CLASS is IN
0x00, 0x00, // TTL (4 bytes) is 32768 second.
0x10, 0x00,
0x00, 0x04, // RDLENGTH is 4
0x01, 0x02, 0x03, 0x04, // 1.2.3.4
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0x05, 'l', 'o', 'c', 'a', 'l',
0x00,
0x00, 0x1C, // Type is AAAA
0x00, 0x01, // CLASS is IN
0x00, 0x00, // TTL (4 bytes) is 32768 second.
0x10, 0x00,
0x00, 0x10, // RDLENGTH is 16
0x01, 0x02, 0x03, 0x04, // 1.2.3.4
0x01, 0x02, 0x03, 0x04,
0x01, 0x02, 0x03, 0x04,
0x01, 0x02, 0x03, 0x04,
};
const uint8_t kGoodbyePacket[] = {
// Header
0x00, 0x00, // ID is zeroed out
0x80, 0x00, // Standard query response, RA, no error
0x00, 0x00, // No questions (for simplicity)
0x00, 0x02, // 1 RR (answers)
0x00, 0x00, // 0 authority RRs
0x00, 0x00, // 0 additional RRs
0x07, '_', 'p', 'r', 'i', 'v', 'e', 't',
0x04, '_', 't', 'c', 'p',
0x05, 'l', 'o', 'c', 'a', 'l',
0x00,
0x00, 0x0c, // TYPE is PTR.
0x00, 0x01, // CLASS is IN.
0x00, 0x00, // TTL (4 bytes) is 0 seconds.
0x00, 0x00,
0x00, 0x0c, // RDLENGTH is 12 bytes.
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0xc0, 0x0c,
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0xc0, 0x0c,
0x00, 0x21, // Type is SRV
0x00, 0x01, // CLASS is IN
0x00, 0x00, // TTL (4 bytes) is 0 seconds.
0x00, 0x00,
0x00, 0x17, // RDLENGTH is 23
0x00, 0x00,
0x00, 0x00,
0x22, 0xb8, // port 8888
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0x05, 'l', 'o', 'c', 'a', 'l',
0x00,
};
const uint8_t kAnnouncePacketRegistered[] = {
// Header
0x00, 0x00, // ID is zeroed out
0x80, 0x00, // Standard query response, RA, no error
0x00, 0x00, // No questions (for simplicity)
0x00, 0x01, // 1 RR (answers)
0x00, 0x00, // 0 authority RRs
0x00, 0x00, // 0 additional RRs
0x09, 'm', 'y', 'S', 'e', 'r', 'v', 'i', 'c', 'e',
0x07, '_', 'p', 'r', 'i', 'v', 'e', 't',
0x04, '_', 't', 'c', 'p',
0x05, 'l', 'o', 'c', 'a', 'l',
0x00,
0x00, 0x10, // TYPE is TXT.
0x00, 0x01, // CLASS is IN.
0x00, 0x00, // TTL (4 bytes) is 32768 seconds.
0x01, 0x00,
0x00, 0x3b, // RDLENGTH is 76 bytes.
0x0a, 'i', 'd', '=', 's', 'o', 'm', 'e', '_', 'i', 'd',
0x10, 't', 'y', '=', 'S', 'a', 'm', 'p', 'l', 'e', ' ',
'd', 'e', 'v', 'i', 'c', 'e',
0x1e, 'n', 'o', 't', 'e', '=',
'S', 'a', 'm', 'p', 'l', 'e', ' ', 'd', 'e', 'v', 'i', 'c', 'e', ' ',
'd', 'e', 's', 'c', 'r', 'i', 'p', 't', 'i', 'o', 'n',
};
const char kResponseInfo[] = "{"
" \"x-privet-token\" : \"MyPrivetToken\""
"}";
const char kResponseInfoWithID[] = "{"
" \"x-privet-token\" : \"MyPrivetToken\","
" \"id\" : \"my_id\""
"}";
const char kResponseRegisterStart[] = "{"
" \"action\": \"start\","
" \"user\": \"user@consumer.example.com\""
"}";
const char kResponseRegisterClaimTokenNoConfirm[] = "{"
" \"action\": \"getClaimToken\","
" \"user\": \"user@consumer.example.com\","
" \"error\": \"pending_user_action\","
" \"timeout\": 1"
"}";
const char kResponseRegisterClaimTokenConfirm[] = "{"
" \"action\": \"getClaimToken\","
" \"user\": \"user@consumer.example.com\","
" \"token\": \"MySampleToken\","
" \"claim_url\": \"http://someurl.com/\""
"}";
const char kResponseCloudPrintConfirm[] = "{ \"success\": true }";
const char kResponseRegisterComplete[] = "{"
" \"action\": \"complete\","
" \"user\": \"user@consumer.example.com\","
" \"device_id\": \"my_id\""
"}";
const char kResponseGaiaToken[] = "{"
" \"access_token\": \"at1\","
" \"expires_in\": 3600,"
" \"token_type\": \"Bearer\""
"}";
const char kResponseGaiaId[] = "{"
" \"id\": \"12345\""
"}";
const char kURLInfo[] = "http://1.2.3.4:8888/privet/info";
const char kURLRegisterStart[] =
"http://1.2.3.4:8888/privet/register?action=start&"
"user=user%40consumer.example.com";
const char kURLRegisterClaimToken[] =
"http://1.2.3.4:8888/privet/register?action=getClaimToken&"
"user=user%40consumer.example.com";
const char kURLCloudPrintConfirm[] =
"https://www.google.com/cloudprint/confirm?token=MySampleToken";
const char kURLRegisterComplete[] =
"http://1.2.3.4:8888/privet/register?action=complete&"
"user=user%40consumer.example.com";
const char kSampleUser[] = "user@consumer.example.com";
void SetURLLoaderFactoryForTest(
Profile* profile,
scoped_refptr<network::SharedURLLoaderFactory> url_loader_factory) {
ChromeSigninClient* signin_client = static_cast<ChromeSigninClient*>(
ChromeSigninClientFactory::GetForProfile(profile));
signin_client->SetURLLoaderFactoryForTest(url_loader_factory);
#if defined(OS_CHROMEOS)
chromeos::AccountManagerFactory* factory =
g_browser_process->platform_part()->GetAccountManagerFactory();
chromeos::AccountManager* account_manager =
factory->GetAccountManager(profile->GetPath().value());
account_manager->SetUrlLoaderFactoryForTests(url_loader_factory);
#endif // defined(OS_CHROMEOS)
}
class TestMessageLoopCondition {
public:
TestMessageLoopCondition() : signaled_(false), waiting_(false) {}
~TestMessageLoopCondition() {}
// Signal a waiting method that it can continue executing.
void Signal() {
signaled_ = true;
if (waiting_)
base::RunLoop::QuitCurrentWhenIdleDeprecated();
}
// Pause execution and recursively run the message loop until |Signal()| is
// called. Do not pause if |Signal()| has already been called.
void Wait() {
while (!signaled_) {
waiting_ = true;
base::RunLoop().Run();
waiting_ = false;
}
signaled_ = false;
}
private:
bool signaled_;
bool waiting_;
DISALLOW_COPY_AND_ASSIGN(TestMessageLoopCondition);
};
class LocalDiscoveryUITest : public WebUIBrowserTest {
public:
LocalDiscoveryUITest()
: set_url_loader_factory_(test_url_loader_factory_.GetSafeWeakWrapper()) {
}
void SetUp() override {
// We need to stub out DualMediaSinkService here, because the profile setup
// instantiates DualMediaSinkService, which in turn sets
// |g_service_discovery_client| with a real instance. This causes
// a DCHECK during TestServiceDiscoveryClient construction.
media_router::DualMediaSinkService::SetInstanceForTest(
new media_router::NoopDualMediaSinkService());
feature_list_.InitAndDisableFeature(media_router::kDialMediaRouteProvider);
WebUIBrowserTest::SetUp();
}
void SetUpOnMainThread() override {
WebUIBrowserTest::SetUpOnMainThread();
#if defined(OS_CHROMEOS)
// On Chrome OS, we need to stub out CupsPrintersManager, because the
// profile setup instatiates a ZeroConfPrinterDetector, which in turn sets
// |g_service_discovery_client| with a real instance. This causes a DCHECK
// during TestServiceDiscoveryClient construction.
chromeos::CupsPrintersManagerFactory::GetInstance()->SetTestingFactory(
ProfileManager::GetActiveUserProfile(),
BrowserContextKeyedServiceFactory::TestingFactory());
#endif
test_service_discovery_client_ = new TestServiceDiscoveryClient();
test_service_discovery_client_->Start();
EXPECT_CALL(
*test_service_discovery_client_.get(),
OnSendTo(std::string((const char*)kQueryData, sizeof(kQueryData))))
.Times(AtLeast(2))
.WillOnce(InvokeWithoutArgs(&condition_devices_listed_,
&TestMessageLoopCondition::Signal))
.WillRepeatedly(Return());
test_url_loader_factory_.AddResponse(
GaiaUrls::GetInstance()->oauth2_token_url().spec(), kResponseGaiaToken);
test_url_loader_factory_.AddResponse(
GaiaUrls::GetInstance()->oauth_user_info_url().spec(), kResponseGaiaId);
test_url_loader_factory_.AddResponse(kURLInfo, kResponseInfo);
test_url_loader_factory_.AddResponse(kURLRegisterStart,
kResponseRegisterStart);
test_url_loader_factory_.AddResponse(kURLRegisterClaimToken,
kResponseRegisterClaimTokenNoConfirm);
test_url_loader_factory_.AddResponse(kURLCloudPrintConfirm,
kResponseCloudPrintConfirm);
test_url_loader_factory_.AddResponse(kURLRegisterComplete,
kResponseRegisterComplete);
identity::MakePrimaryAccountAvailable(
IdentityManagerFactory::GetForProfile(browser()->profile()),
kSampleUser);
AddLibrary(base::FilePath(FILE_PATH_LITERAL("local_discovery_ui_test.js")));
SetURLLoaderFactoryForTest(
ProfileManager::GetActiveUserProfile() /* profile */,
test_url_loader_factory_.GetSafeWeakWrapper());
}
void TearDownOnMainThread() override {
test_service_discovery_client_ = nullptr;
WebUIBrowserTest::TearDownOnMainThread();
}
void SetUpCommandLine(base::CommandLine* command_line) override {
#if defined(OS_CHROMEOS)
// On chromeos, don't sign in with the stub-user automatically. Use the
// kLoginUser instead.
command_line->AppendSwitchASCII(chromeos::switches::kLoginUser,
kSampleUser);
command_line->AppendSwitchASCII(chromeos::switches::kLoginProfile,
chrome::kTestUserProfileDir);
#endif
command_line->AppendSwitch(switches::kDisableDeviceDiscoveryNotifications);
WebUIBrowserTest::SetUpCommandLine(command_line);
}
void RunFor(base::TimeDelta time_period) {
base::CancelableCallback<void()> callback(
base::Bind(&base::RunLoop::QuitCurrentWhenIdleDeprecated));
base::ThreadTaskRunnerHandle::Get()->PostDelayedTask(
FROM_HERE, callback.callback(), time_period);
base::RunLoop().Run();
callback.Cancel();
}
TestServiceDiscoveryClient* test_service_discovery_client() {
return test_service_discovery_client_.get();
}
TestMessageLoopCondition& condition_devices_listed() {
return condition_devices_listed_;
}
network::TestURLLoaderFactory* test_url_loader_factory() {
return &test_url_loader_factory_;
}
private:
scoped_refptr<TestServiceDiscoveryClient> test_service_discovery_client_;
TestMessageLoopCondition condition_devices_listed_;
network::TestURLLoaderFactory test_url_loader_factory_;
local_discovery::LocalDiscoveryUIHandler::SetURLLoaderFactoryForTesting
set_url_loader_factory_;
base::test::ScopedFeatureList feature_list_;
DISALLOW_COPY_AND_ASSIGN(LocalDiscoveryUITest);
};
IN_PROC_BROWSER_TEST_F(LocalDiscoveryUITest, EmptyTest) {
ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUIDevicesURL));
condition_devices_listed().Wait();
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("checkNoDevices"));
}
IN_PROC_BROWSER_TEST_F(LocalDiscoveryUITest, AddRowTest) {
ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUIDevicesURL));
condition_devices_listed().Wait();
test_service_discovery_client()->SimulateReceive(kAnnouncePacket,
sizeof(kAnnouncePacket));
base::RunLoop().RunUntilIdle();
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("checkOneDevice"));
test_service_discovery_client()->SimulateReceive(kGoodbyePacket,
sizeof(kGoodbyePacket));
RunFor(base::TimeDelta::FromMilliseconds(1100));
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("checkNoDevices"));
}
IN_PROC_BROWSER_TEST_F(LocalDiscoveryUITest, RegisterTest) {
ui_test_utils::NavigateToURL(browser(), GURL(chrome::kChromeUIDevicesURL));
condition_devices_listed().Wait();
test_service_discovery_client()->SimulateReceive(kAnnouncePacket,
sizeof(kAnnouncePacket));
base::RunLoop().RunUntilIdle();
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("checkOneDevice"));
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("registerShowOverlay"));
{
base::RunLoop run_loop;
std::set<GURL> served_urls;
test_url_loader_factory()->SetInterceptor(base::BindLambdaForTesting(
[&](const network::ResourceRequest& resource) {
served_urls.insert(resource.url);
if (resource.url == GURL(kURLRegisterClaimToken))
run_loop.Quit();
}));
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("registerBegin"));
run_loop.Run();
EXPECT_TRUE(base::Contains(served_urls, GURL(kURLInfo)));
EXPECT_TRUE(base::Contains(served_urls, GURL(kURLRegisterStart)));
EXPECT_TRUE(base::Contains(served_urls, GURL(kURLRegisterClaimToken)));
test_url_loader_factory()->SetInterceptor(base::NullCallback());
}
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("expectPageAdding1"));
test_url_loader_factory()->AddResponse(kURLRegisterClaimToken,
kResponseRegisterClaimTokenConfirm);
test_url_loader_factory()->AddResponse(kURLInfo, kResponseInfoWithID);
{
base::RunLoop run_loop;
std::set<GURL> served_urls;
test_url_loader_factory()->SetInterceptor(base::BindLambdaForTesting(
[&](const network::ResourceRequest& resource) {
served_urls.insert(resource.url);
if (resource.url == GURL(kURLInfo))
run_loop.Quit();
}));
run_loop.Run();
EXPECT_TRUE(base::Contains(served_urls, GURL(kURLRegisterClaimToken)));
EXPECT_TRUE(base::Contains(served_urls, GURL(kURLCloudPrintConfirm)));
EXPECT_TRUE(base::Contains(served_urls, GURL(kURLRegisterComplete)));
EXPECT_TRUE(base::Contains(served_urls, GURL(kURLInfo)));
test_url_loader_factory()->SetInterceptor(base::NullCallback());
}
test_service_discovery_client()->SimulateReceive(
kAnnouncePacketRegistered, sizeof(kAnnouncePacketRegistered));
base::RunLoop().RunUntilIdle();
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("expectRegisterDone"));
}
} // namespace
} // namespace local_discovery