| // Copyright 2014 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/ash/app_mode/fake_cws.h" |
| |
| #include <utility> |
| |
| #include "base/bind.h" |
| #include "base/command_line.h" |
| #include "base/files/file_util.h" |
| #include "base/path_service.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "base/strings/string_tokenizer.h" |
| #include "base/strings/string_util.h" |
| #include "base/threading/thread_restrictions.h" |
| #include "chrome/common/chrome_paths.h" |
| #include "chrome/common/chrome_switches.h" |
| #include "chrome/common/initialize_extensions_client.h" |
| #include "chrome/test/base/in_process_browser_test.h" |
| #include "crypto/sha2.h" |
| #include "extensions/common/extensions_client.h" |
| #include "net/base/url_util.h" |
| #include "net/test/embedded_test_server/embedded_test_server.h" |
| #include "net/test/embedded_test_server/http_request.h" |
| |
| using net::test_server::BasicHttpResponse; |
| using net::test_server::HttpRequest; |
| using net::test_server::HttpResponse; |
| |
| namespace ash { |
| |
| namespace { |
| |
| const char kWebstoreDomain[] = "cws.com"; |
| // Kiosk app crx file download path under web store site. |
| const char kCrxDownloadPath[] = "/chromeos/app_mode/webstore/downloads/"; |
| |
| const char kAppNoUpdateTemplate[] = |
| "<app appid=\"$AppId\" status=\"ok\">" |
| "<updatecheck status=\"noupdate\"/>" |
| "</app>"; |
| |
| const char kAppHasUpdateTemplate[] = |
| "<app appid=\"$AppId\" status=\"ok\">" |
| "<updatecheck codebase=\"$CrxDownloadUrl\" fp=\"1.$FP\" " |
| "hash=\"\" hash_sha256=\"$FP\" size=\"$Size\" status=\"ok\" " |
| "version=\"$Version\"/>" |
| "</app>"; |
| |
| const char kPrivateStoreAppHasUpdateTemplate[] = |
| "<app appid=\"$AppId\">" |
| "<updatecheck codebase=\"$CrxDownloadUrl\" version=\"$Version\"/>" |
| "</app>"; |
| |
| const char kUpdateContentTemplate[] = |
| "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" |
| "<gupdate xmlns=\"http://www.google.com/update2/response\" " |
| "protocol=\"2.0\" server=\"prod\">" |
| "<daystart elapsed_days=\"2569\" elapsed_seconds=\"36478\"/>" |
| "$APPS" |
| "</gupdate>"; |
| |
| const char kAppNoUpdateTemplateJSON[] = |
| "{\"appid\": \"$AppId\"," |
| " \"status\": \"ok\"," |
| " \"updatecheck\": { \"status\": \"noupdate\" }" |
| "}"; |
| |
| const char kAppHasUpdateTemplateJSON[] = |
| "{" |
| " \"appid\": \"$AppId\"," |
| " \"status\": \"ok\"," |
| " \"updatecheck\": {" |
| " \"status\": \"ok\"," |
| " \"manifest\": {" |
| " \"version\": \"$Version\"," |
| " \"packages\": {" |
| " \"package\": [" |
| " {" |
| " \"fp\": \"1.$FP\"," |
| " \"size\": \"$Size\"," |
| " \"hash_sha256\": \"$FP\"," |
| " \"name\": \"\"" |
| " }" |
| " ]" |
| " }" |
| " }," |
| " \"urls\": { \"url\": [ { \"codebase\": \"$CrxDownloadUrl\"} ] }" |
| " }" |
| "}"; |
| |
| const char kUpdateContentTemplateJSON[] = |
| ")]}'\n" |
| "{" |
| " \"response\": {" |
| " \"protocol\": \"3.1\"," |
| " \"daystart\": {" |
| " \"elapsed_days\": 2569," |
| " \"elapsed_seconds\": 36478" |
| " }," |
| " \"app\": [" |
| " $APPS" |
| " ]" |
| " }" |
| "}"; |
| |
| const char kAppIdHeader[] = "X-Goog-Update-AppId"; |
| |
| bool GetAppIdsFromHeader(const HttpRequest::HeaderMap& headers, |
| std::vector<std::string>* ids) { |
| if (headers.count(kAppIdHeader) == 0) |
| return false; |
| base::StringTokenizer t(headers.at(kAppIdHeader), ","); |
| while (t.GetNext()) { |
| ids->push_back(t.token()); |
| } |
| return !ids->empty(); |
| } |
| |
| bool GetAppIdsFromUpdateUrl(const GURL& update_url, |
| std::vector<std::string>* ids) { |
| for (net::QueryIterator it(update_url); !it.IsAtEnd(); it.Advance()) { |
| if (it.GetKey() != "x") |
| continue; |
| std::string id; |
| net::GetValueForKeyInQuery(GURL("http://dummy?" + it.GetUnescapedValue()), |
| "id", &id); |
| ids->push_back(id); |
| } |
| return !ids->empty(); |
| } |
| |
| // FakeCWS uses ScopedIgnoreContentVerifierForTest to disable extension |
| // content verification. This helper could be instantiated only once. Usually |
| // that not an issue, since FakeCWS is also instantiated only once, in a base |
| // test class. Some tests use a secondary FakeCWS instance. This flag will be |
| // set by first created FakeCWS (which we'll call "primary"), and only primary |
| // FakeCWS will hold the ScopedIgnoreContentVerifierForTest instance. |
| bool g_is_fakecws_active = false; |
| |
| std::string ApplyHasNoUpdateTemplate(std::string app_id, |
| bool use_json, |
| bool use_private_store) { |
| std::string update_check_content(use_json ? kAppNoUpdateTemplateJSON |
| : kAppNoUpdateTemplate); |
| base::ReplaceSubstringsAfterOffset(&update_check_content, 0, "$AppId", |
| app_id); |
| return update_check_content; |
| } |
| |
| std::string ApplyHasUpdateTemplate(std::string app_id, |
| GURL download_url, |
| std::string sha256_hex, |
| int size, |
| std::string version, |
| bool use_json, |
| bool use_private_store) { |
| std::string update_check_content( |
| use_json ? kAppHasUpdateTemplateJSON |
| : use_private_store ? kPrivateStoreAppHasUpdateTemplate |
| : kAppHasUpdateTemplate); |
| base::ReplaceSubstringsAfterOffset(&update_check_content, 0, "$AppId", |
| app_id); |
| base::ReplaceSubstringsAfterOffset(&update_check_content, 0, |
| "$CrxDownloadUrl", download_url.spec()); |
| base::ReplaceSubstringsAfterOffset(&update_check_content, 0, "$FP", |
| sha256_hex); |
| base::ReplaceSubstringsAfterOffset(&update_check_content, 0, "$Size", |
| base::NumberToString(size)); |
| base::ReplaceSubstringsAfterOffset(&update_check_content, 0, "$Version", |
| version); |
| return update_check_content; |
| } |
| |
| } // namespace |
| |
| FakeCWS::FakeCWS() : update_check_count_(0) { |
| if (!g_is_fakecws_active) { |
| g_is_fakecws_active = true; |
| scoped_ignore_content_verifier_ = |
| std::make_unique<extensions::ScopedIgnoreContentVerifierForTest>(); |
| } |
| } |
| |
| FakeCWS::~FakeCWS() { |
| // If the secondary FakeCWS was desructed after primary one, secondary will |
| // work without scoped_ignore_content_verifier_. We want to catch such a |
| // situation, so we check that primary FakeCWS is not destroyed yet. |
| DCHECK(g_is_fakecws_active); |
| |
| if (scoped_ignore_content_verifier_) |
| g_is_fakecws_active = false; |
| } |
| |
| void FakeCWS::Init(net::EmbeddedTestServer* embedded_test_server) { |
| use_private_store_templates_ = false; |
| update_check_end_point_ = "/update_check.xml"; |
| |
| SetupWebStoreURL(embedded_test_server->base_url()); |
| OverrideGalleryCommandlineSwitches(); |
| embedded_test_server->RegisterRequestHandler( |
| base::BindRepeating(&FakeCWS::HandleRequest, base::Unretained(this))); |
| } |
| |
| void FakeCWS::InitAsPrivateStore(net::EmbeddedTestServer* embedded_test_server, |
| const std::string& update_check_end_point) { |
| use_private_store_templates_ = true; |
| update_check_end_point_ = update_check_end_point; |
| |
| SetupWebStoreURL(embedded_test_server->base_url()); |
| OverrideGalleryCommandlineSwitches(); |
| |
| embedded_test_server->RegisterRequestHandler( |
| base::BindRepeating(&FakeCWS::HandleRequest, base::Unretained(this))); |
| } |
| |
| void FakeCWS::SetUpdateCrx(const std::string& app_id, |
| const std::string& crx_file, |
| const std::string& version) { |
| GURL crx_download_url = web_store_url_.Resolve(kCrxDownloadPath + crx_file); |
| |
| base::FilePath test_data_dir; |
| base::PathService::Get(chrome::DIR_TEST_DATA, &test_data_dir); |
| base::FilePath crx_file_path = |
| test_data_dir.AppendASCII("chromeos/app_mode/webstore/downloads") |
| .AppendASCII(crx_file); |
| std::string crx_content; |
| { |
| base::ScopedAllowBlockingForTesting allow_io; |
| ASSERT_TRUE(base::ReadFileToString(crx_file_path, &crx_content)); |
| } |
| |
| const std::string sha256 = crypto::SHA256HashString(crx_content); |
| const std::string sha256_hex = base::HexEncode(sha256.c_str(), sha256.size()); |
| |
| id_to_update_check_content_map_[app_id] = |
| base::BindRepeating(&ApplyHasUpdateTemplate, app_id, crx_download_url, |
| sha256_hex, crx_content.size(), version); |
| } |
| |
| void FakeCWS::SetNoUpdate(const std::string& app_id) { |
| id_to_update_check_content_map_[app_id] = |
| base::BindRepeating(&ApplyHasNoUpdateTemplate, app_id); |
| } |
| |
| int FakeCWS::GetUpdateCheckCountAndReset() { |
| int current_count = update_check_count_; |
| update_check_count_ = 0; |
| return current_count; |
| } |
| |
| void FakeCWS::SetupWebStoreURL(const GURL& test_server_url) { |
| GURL::Replacements replace_webstore_host; |
| replace_webstore_host.SetHostStr(kWebstoreDomain); |
| web_store_url_ = test_server_url.ReplaceComponents(replace_webstore_host); |
| } |
| |
| void FakeCWS::OverrideGalleryCommandlineSwitches() { |
| DCHECK(web_store_url_.is_valid()); |
| |
| base::CommandLine* command_line = base::CommandLine::ForCurrentProcess(); |
| |
| command_line->AppendSwitchASCII( |
| ::switches::kAppsGalleryURL, |
| web_store_url_.Resolve("/chromeos/app_mode/webstore").spec()); |
| |
| std::string downloads_path = std::string(kCrxDownloadPath).append("%s.crx"); |
| GURL downloads_url = web_store_url_.Resolve(downloads_path); |
| command_line->AppendSwitchASCII(::switches::kAppsGalleryDownloadURL, |
| downloads_url.spec()); |
| |
| GURL update_url = web_store_url_.Resolve(update_check_end_point_); |
| command_line->AppendSwitchASCII(::switches::kAppsGalleryUpdateURL, |
| update_url.spec()); |
| |
| EnsureExtensionsClientInitialized(); |
| extensions::ExtensionsClient::Get()->InitializeWebStoreUrls(command_line); |
| } |
| |
| bool FakeCWS::GetUpdateCheckContent(const std::vector<std::string>& ids, |
| std::string* update_check_content, |
| bool use_json) { |
| std::string apps_content; |
| bool need_comma = false; |
| for (const std::string& id : ids) { |
| std::string app_update_content; |
| auto it = id_to_update_check_content_map_.find(id); |
| if (it == id_to_update_check_content_map_.end()) |
| return false; |
| if (need_comma) |
| apps_content.append(","); |
| apps_content.append(it->second.Run(use_json, use_private_store_templates_)); |
| need_comma = use_json; |
| } |
| if (apps_content.empty()) |
| return false; |
| |
| *update_check_content = |
| use_json ? kUpdateContentTemplateJSON : kUpdateContentTemplate; |
| base::ReplaceSubstringsAfterOffset(update_check_content, 0, "$APPS", |
| apps_content); |
| return true; |
| } |
| |
| std::unique_ptr<HttpResponse> FakeCWS::HandleRequest( |
| const HttpRequest& request) { |
| GURL request_url = GURL("http://localhost").Resolve(request.relative_url); |
| std::string request_path = request_url.path(); |
| if (request_path.find(update_check_end_point_) != std::string::npos && |
| !id_to_update_check_content_map_.empty()) { |
| std::vector<std::string> ids; |
| if (GetAppIdsFromHeader(request.headers, &ids) || |
| GetAppIdsFromUpdateUrl(request_url, &ids)) { |
| bool use_json = |
| request.content.size() > 0 && request.content.at(0) == '{'; |
| std::string update_check_content; |
| if (GetUpdateCheckContent(ids, &update_check_content, use_json)) { |
| ++update_check_count_; |
| std::unique_ptr<BasicHttpResponse> http_response( |
| new BasicHttpResponse()); |
| http_response->set_code(net::HTTP_OK); |
| if (!use_json) |
| http_response->set_content_type("text/xml"); |
| http_response->set_content(update_check_content); |
| return std::move(http_response); |
| } |
| } |
| } |
| |
| return nullptr; |
| } |
| |
| } // namespace ash |