blob: 42a5d2e441445551e3ff4a0a81d4ddf99345a368 [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 "base/bind.h"
#include "base/file_util.h"
#include "base/files/file_path.h"
#include "base/memory/ref_counted.h"
#include "base/memory/scoped_ptr.h"
#include "base/path_service.h"
#include "base/run_loop.h"
#include "base/threading/sequenced_worker_pool.h"
#include "chrome/browser/component_updater/crx_downloader.h"
#include "chrome/common/chrome_paths.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/test/test_browser_thread_bundle.h"
#include "content/test/net/url_request_prepackaged_interceptor.h"
#include "net/base/net_errors.h"
#include "net/url_request/url_request_test_util.h"
#include "testing/gtest/include/gtest/gtest.h"
using content::BrowserThread;
namespace component_updater {
namespace {
// Intercepts HTTP GET requests sent to "localhost".
typedef content::URLLocalHostRequestPrepackagedInterceptor GetInterceptor;
base::FilePath test_file(const char* file) {
base::FilePath path;
PathService::Get(chrome::DIR_TEST_DATA, &path);
return path.AppendASCII("components").AppendASCII(file);
}
} // namespace
class CrxDownloaderTest : public testing::Test {
public:
CrxDownloaderTest();
virtual ~CrxDownloaderTest();
// Overrides from testing::Test.
virtual void SetUp() OVERRIDE;
virtual void TearDown() OVERRIDE;
void Quit();
void RunThreads();
void RunThreadsUntilIdle();
void DownloadComplete(int crx_context, const CrxDownloader::Result& result);
protected:
scoped_ptr<CrxDownloader> crx_downloader_;
int crx_context_;
int error_;
base::FilePath response_;
int num_calls_;
// A magic value for the context to be used in the tests.
static const int kExpectedContext = 0xaabb;
private:
scoped_refptr<base::SequencedTaskRunner> blocking_task_runner_;
scoped_refptr<net::TestURLRequestContextGetter> context_;
content::TestBrowserThreadBundle thread_bundle_;
base::Closure quit_closure_;
};
const int CrxDownloaderTest::kExpectedContext;
CrxDownloaderTest::CrxDownloaderTest()
: crx_context_(0),
error_(0),
num_calls_(0),
blocking_task_runner_(BrowserThread::GetBlockingPool()->
GetSequencedTaskRunnerWithShutdownBehavior(
BrowserThread::GetBlockingPool()->GetSequenceToken(),
base::SequencedWorkerPool::SKIP_ON_SHUTDOWN)),
context_(new net::TestURLRequestContextGetter(
BrowserThread::GetMessageLoopProxyForThread(BrowserThread::IO))),
thread_bundle_(content::TestBrowserThreadBundle::IO_MAINLOOP) {
}
CrxDownloaderTest::~CrxDownloaderTest() {
context_ = NULL;
}
void CrxDownloaderTest::SetUp() {
num_calls_ = 0;
crx_downloader_.reset(CrxDownloader::Create(
false, // Do not use the background downloader in these tests.
context_.get(),
blocking_task_runner_,
base::Bind(&CrxDownloaderTest::DownloadComplete,
base::Unretained(this),
kExpectedContext)));
}
void CrxDownloaderTest::TearDown() {
}
void CrxDownloaderTest::Quit() {
if (!quit_closure_.is_null())
quit_closure_.Run();
}
void CrxDownloaderTest::DownloadComplete(
int crx_context,
const CrxDownloader::Result& result) {
++num_calls_;
crx_context_ = crx_context;
error_ = result.error;
response_ = result.response;
Quit();
}
void CrxDownloaderTest::RunThreads() {
base::RunLoop runloop;
quit_closure_ = runloop.QuitClosure();
runloop.Run();
// Since some tests need to drain currently enqueued tasks such as network
// intercepts on the IO thread, run the threads until they are
// idle. The component updater service won't loop again until the loop count
// is set and the service is started.
RunThreadsUntilIdle();
}
void CrxDownloaderTest::RunThreadsUntilIdle() {
base::RunLoop().RunUntilIdle();
}
// Tests that starting a download without a url results in an error.
TEST_F(CrxDownloaderTest, NoUrl) {
std::vector<GURL> urls;
crx_downloader_->StartDownload(urls);
RunThreadsUntilIdle();
EXPECT_EQ(-1, error_);
EXPECT_EQ(kExpectedContext, crx_context_);
EXPECT_EQ(1, num_calls_);
}
// Tests that downloading from one url is successful.
TEST_F(CrxDownloaderTest, OneUrl) {
const GURL expected_crx_url =
GURL("http://localhost/download/jebgalgnebhfojomionfpkfelancnnkf.crx");
GetInterceptor interceptor;
interceptor.SetResponse(
expected_crx_url,
test_file("jebgalgnebhfojomionfpkfelancnnkf.crx"));
crx_downloader_->StartDownloadFromUrl(expected_crx_url);
RunThreads();
EXPECT_EQ(1, interceptor.GetHitCount());
EXPECT_EQ(0, error_);
EXPECT_EQ(kExpectedContext, crx_context_);
EXPECT_EQ(1, num_calls_);
}
// Tests that specifying from two urls has no side effects. Expect a successful
// download, and only one download request be made.
TEST_F(CrxDownloaderTest, TwoUrls) {
const GURL expected_crx_url =
GURL("http://localhost/download/jebgalgnebhfojomionfpkfelancnnkf.crx");
GetInterceptor interceptor;
interceptor.SetResponse(
expected_crx_url,
test_file("jebgalgnebhfojomionfpkfelancnnkf.crx"));
std::vector<GURL> urls;
urls.push_back(expected_crx_url);
urls.push_back(expected_crx_url);
crx_downloader_->StartDownload(urls);
RunThreads();
EXPECT_EQ(1, interceptor.GetHitCount());
EXPECT_EQ(0, error_);
EXPECT_EQ(kExpectedContext, crx_context_);
EXPECT_EQ(1, num_calls_);
}
// Tests that an invalid host results in a download error.
TEST_F(CrxDownloaderTest, OneUrl_InvalidHost) {
const GURL expected_crx_url =
GURL("http://localhost/download/jebgalgnebhfojomionfpkfelancnnkf.crx");
GetInterceptor interceptor;
interceptor.SetResponse(
expected_crx_url,
test_file("jebgalgnebhfojomionfpkfelancnnkf.crx"));
crx_downloader_->StartDownloadFromUrl(
GURL("http://no.such.host"
"/download/jebgalgnebhfojomionfpkfelancnnkf.crx"));
RunThreads();
EXPECT_EQ(0, interceptor.GetHitCount());
EXPECT_NE(0, error_);
EXPECT_EQ(kExpectedContext, crx_context_);
EXPECT_EQ(1, num_calls_);
}
// Tests that an invalid path results in a download error.
TEST_F(CrxDownloaderTest, OneUrl_InvalidPath) {
const GURL expected_crx_url =
GURL("http://localhost/download/jebgalgnebhfojomionfpkfelancnnkf.crx");
GetInterceptor interceptor;
interceptor.SetResponse(
expected_crx_url,
test_file("jebgalgnebhfojomionfpkfelancnnkf.crx"));
crx_downloader_->StartDownloadFromUrl(GURL("http://localhost/no/such/file"));
RunThreads();
EXPECT_EQ(0, interceptor.GetHitCount());
EXPECT_NE(0, error_);
EXPECT_EQ(kExpectedContext, crx_context_);
EXPECT_EQ(1, num_calls_);
}
// Tests that the fallback to a valid url is successful.
TEST_F(CrxDownloaderTest, TwoUrls_FirstInvalid) {
const GURL expected_crx_url =
GURL("http://localhost/download/jebgalgnebhfojomionfpkfelancnnkf.crx");
GetInterceptor interceptor;
interceptor.SetResponse(
expected_crx_url,
test_file("jebgalgnebhfojomionfpkfelancnnkf.crx"));
std::vector<GURL> urls;
urls.push_back(GURL("http://localhost/no/such/file"));
urls.push_back(expected_crx_url);
crx_downloader_->StartDownload(urls);
RunThreads();
EXPECT_EQ(1, interceptor.GetHitCount());
EXPECT_EQ(0, error_);
EXPECT_EQ(kExpectedContext, crx_context_);
EXPECT_EQ(1, num_calls_);
}
// Tests that the download succeeds if the first url is correct and the
// second bad url does not have a side-effect.
TEST_F(CrxDownloaderTest, TwoUrls_SecondInvalid) {
const GURL expected_crx_url =
GURL("http://localhost/download/jebgalgnebhfojomionfpkfelancnnkf.crx");
GetInterceptor interceptor;
interceptor.SetResponse(
expected_crx_url,
test_file("jebgalgnebhfojomionfpkfelancnnkf.crx"));
std::vector<GURL> urls;
urls.push_back(expected_crx_url);
urls.push_back(GURL("http://localhost/no/such/file"));
crx_downloader_->StartDownload(urls);
RunThreads();
EXPECT_EQ(1, interceptor.GetHitCount());
EXPECT_EQ(0, error_);
EXPECT_EQ(kExpectedContext, crx_context_);
EXPECT_EQ(1, num_calls_);
}
// Tests that the download fails if both urls are bad.
TEST_F(CrxDownloaderTest, TwoUrls_BothInvalid) {
const GURL expected_crx_url =
GURL("http://localhost/download/jebgalgnebhfojomionfpkfelancnnkf.crx");
GetInterceptor interceptor;
interceptor.SetResponse(
expected_crx_url,
test_file("jebgalgnebhfojomionfpkfelancnnkf.crx"));
std::vector<GURL> urls;
urls.push_back(GURL("http://localhost/no/such/file"));
urls.push_back(GURL("http://no.such.host/"
"/download/jebgalgnebhfojomionfpkfelancnnkf.crx"));
crx_downloader_->StartDownload(urls);
RunThreads();
EXPECT_EQ(0, interceptor.GetHitCount());
EXPECT_NE(0, error_);
EXPECT_EQ(kExpectedContext, crx_context_);
EXPECT_EQ(1, num_calls_);
}
} // namespace component_updater