blob: 2488a4195d151d07f13b204da3711795f400195f [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/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/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/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/signin/profile_oauth2_token_service_factory.h"
#include "chrome/browser/signin/signin_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 "net/url_request/test_url_fetcher_factory.h"
#include "net/url_request/url_request_status.h"
#include "net/url_request/url_request_test_util.h"
#include "services/identity/public/cpp/identity_test_utils.h"
#include "services/network/public/cpp/shared_url_loader_factory.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/common/pref_names.h"
#include "chromeos/chromeos_switches.h"
#include "components/prefs/pref_service.h"
#endif
using testing::InvokeWithoutArgs;
using testing::Return;
using testing::AtLeast;
using testing::DoDefault;
using testing::DoAll;
using testing::InSequence;
using testing::StrictMock;
using testing::AnyNumber;
using testing::InvokeWithoutArgs;
using testing::Return;
using testing::AtLeast;
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";
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 MockableFakeURLFetcherCreator {
public:
MockableFakeURLFetcherCreator() {
}
~MockableFakeURLFetcherCreator() {
}
MOCK_METHOD1(OnCreateFakeURLFetcher, void(const std::string& url));
std::unique_ptr<net::FakeURLFetcher> CreateFakeURLFetcher(
const GURL& url,
net::URLFetcherDelegate* delegate,
const std::string& response_data,
net::HttpStatusCode response_code,
net::URLRequestStatus::Status status) {
OnCreateFakeURLFetcher(url.spec());
return std::unique_ptr<net::FakeURLFetcher>(new net::FakeURLFetcher(
url, delegate, response_data, response_code, status));
}
net::FakeURLFetcherFactory::FakeURLFetcherCreator callback() {
return base::Bind(&MockableFakeURLFetcherCreator::CreateFakeURLFetcher,
base::Unretained(this));
}
};
class LocalDiscoveryUITest : public WebUIBrowserTest {
public:
LocalDiscoveryUITest()
: test_shared_loader_factory_(
base::MakeRefCounted<network::WeakWrapperSharedURLLoaderFactory>(
&test_url_loader_factory_)),
fake_fetcher_factory_(&fetcher_impl_factory_,
fake_url_fetcher_creator_.callback()) {}
~LocalDiscoveryUITest() override {
}
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());
WebUIBrowserTest::SetUp();
}
void SetUpOnMainThread() override {
WebUIBrowserTest::SetUpOnMainThread();
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);
fake_fetcher_factory().SetFakeResponse(
GURL(kURLInfo),
kResponseInfo,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
fake_fetcher_factory().SetFakeResponse(
GURL(kURLRegisterStart),
kResponseRegisterStart,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
fake_fetcher_factory().SetFakeResponse(
GURL(kURLRegisterClaimToken),
kResponseRegisterClaimTokenNoConfirm,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
fake_fetcher_factory().SetFakeResponse(
GURL(kURLCloudPrintConfirm),
kResponseCloudPrintConfirm,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
fake_fetcher_factory().SetFakeResponse(
GURL(kURLRegisterComplete),
kResponseRegisterComplete,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
fake_fetcher_factory().SetFakeResponse(
GaiaUrls::GetInstance()->oauth2_token_url(),
kResponseGaiaToken,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(
GaiaUrls::GetInstance()->oauth2_token_url().spec()))
.Times(AnyNumber());
fake_fetcher_factory().SetFakeResponse(
GaiaUrls::GetInstance()->oauth_user_info_url(),
kResponseGaiaId,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(
GaiaUrls::GetInstance()->oauth_user_info_url().spec()))
.Times(AnyNumber());
identity::MakePrimaryAccountAvailable(
SigninManagerFactory::GetForProfile(browser()->profile()),
ProfileOAuth2TokenServiceFactory::GetForProfile(browser()->profile()),
IdentityManagerFactory::GetForProfile(browser()->profile()),
kSampleUser);
AddLibrary(base::FilePath(FILE_PATH_LITERAL("local_discovery_ui_test.js")));
ChromeSigninClient* signin_client = static_cast<ChromeSigninClient*>(
ChromeSigninClientFactory::GetForProfile(
ProfileManager::GetActiveUserProfile()));
signin_client->SetURLLoaderFactoryForTest(test_shared_loader_factory_);
}
void TearDownOnMainThread() override {
test_service_discovery_client_ = nullptr;
test_shared_loader_factory_->Detach();
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_;
}
net::FakeURLFetcherFactory& fake_fetcher_factory() {
return fake_fetcher_factory_;
}
MockableFakeURLFetcherCreator& fake_url_fetcher_creator() {
return fake_url_fetcher_creator_;
}
private:
scoped_refptr<TestServiceDiscoveryClient> test_service_discovery_client_;
TestMessageLoopCondition condition_devices_listed_;
network::TestURLLoaderFactory test_url_loader_factory_;
scoped_refptr<network::WeakWrapperSharedURLLoaderFactory>
test_shared_loader_factory_;
net::URLFetcherImplFactory fetcher_impl_factory_;
StrictMock<MockableFakeURLFetcherCreator> fake_url_fetcher_creator_;
net::FakeURLFetcherFactory fake_fetcher_factory_;
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"));
}
// Flaky: http://crbug.com/660669.
IN_PROC_BROWSER_TEST_F(LocalDiscoveryUITest, RegisterTest) {
TestMessageLoopCondition condition_token_claimed;
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"));
{
InSequence s;
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(kURLInfo));
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(
kURLRegisterStart));
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(
kURLRegisterClaimToken))
.WillOnce(InvokeWithoutArgs(&condition_token_claimed,
&TestMessageLoopCondition::Signal));
}
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("registerBegin"));
condition_token_claimed.Wait();
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("expectPageAdding1"));
fake_fetcher_factory().SetFakeResponse(
GURL(kURLRegisterClaimToken),
kResponseRegisterClaimTokenConfirm,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
fake_fetcher_factory().SetFakeResponse(
GURL(kURLInfo),
kResponseInfoWithID,
net::HTTP_OK,
net::URLRequestStatus::SUCCESS);
{
InSequence s;
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(
kURLRegisterClaimToken));
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(
kURLCloudPrintConfirm));
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(
kURLRegisterComplete));
EXPECT_CALL(fake_url_fetcher_creator(), OnCreateFakeURLFetcher(kURLInfo))
.WillOnce(InvokeWithoutArgs(&condition_token_claimed,
&TestMessageLoopCondition::Signal));
}
condition_token_claimed.Wait();
test_service_discovery_client()->SimulateReceive(
kAnnouncePacketRegistered, sizeof(kAnnouncePacketRegistered));
base::RunLoop().RunUntilIdle();
EXPECT_TRUE(WebUIBrowserTest::RunJavascriptTest("expectRegisterDone"));
}
} // namespace
} // namespace local_discovery