blob: 30753ea2d82f602fa0396a89111ac615460894fa [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/renderer/chrome_content_renderer_client.h"
#include <stddef.h>
#include <map>
#include <memory>
#include <string>
#include <utility>
#include <vector>
#include "base/command_line.h"
#include "base/strings/utf_string_conversions.h"
#include "base/values.h"
#include "build/build_config.h"
#include "chrome/common/privacy_budget/scoped_privacy_budget_config.h"
#include "content/public/common/content_switches.h"
#include "content/public/common/webplugininfo.h"
#include "extensions/buildflags/buildflags.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/blink/public/platform/web_string.h"
#include "third_party/blink/public/platform/web_url_response.h"
#include "url/gurl.h"
#if BUILDFLAG(ENABLE_EXTENSIONS)
#include "chrome/common/extensions/extension_test_util.h"
#include "extensions/common/extension.h"
#include "extensions/common/extension_builder.h"
#include "extensions/common/extensions_client.h"
#include "extensions/common/manifest_constants.h"
#endif
#if BUILDFLAG(ENABLE_NACL)
#include "third_party/blink/public/platform/web_vector.h"
#include "third_party/blink/public/web/web_plugin_params.h"
#endif
#if BUILDFLAG(ENABLE_NACL)
using blink::WebPluginParams;
using blink::WebString;
using blink::WebVector;
#endif
using content::WebPluginInfo;
using content::WebPluginMimeType;
#if BUILDFLAG(ENABLE_EXTENSIONS)
using extensions::mojom::ManifestLocation;
#endif
namespace {
#if BUILDFLAG(ENABLE_NACL)
const bool kNaClRestricted = false;
const bool kNaClUnrestricted = true;
const bool kExtensionNotFromWebStore = false;
const bool kExtensionFromWebStore = true;
#endif
#if BUILDFLAG(ENABLE_EXTENSIONS)
const bool kNotHostedApp = false;
const bool kHostedApp = true;
#endif
#if BUILDFLAG(ENABLE_NACL)
const char kExtensionUrl[] = "chrome-extension://extension_id/background.html";
#endif
void AddContentTypeHandler(content::WebPluginInfo* info,
const char* mime_type,
const char* manifest_url) {
content::WebPluginMimeType mime_type_info;
mime_type_info.mime_type = mime_type;
mime_type_info.additional_params.emplace_back(
u"nacl", base::UTF8ToUTF16(manifest_url));
info->mime_types.push_back(mime_type_info);
}
} // namespace
class ChromeContentRendererClientTest : public testing::Test {
public:
void SetUp() override {
// Ensure that this looks like the renderer process based on the command
// line.
base::CommandLine::ForCurrentProcess()->AppendSwitchASCII(
switches::kProcessType, switches::kRendererProcess);
}
};
#if BUILDFLAG(ENABLE_EXTENSIONS)
scoped_refptr<const extensions::Extension> CreateTestExtension(
ManifestLocation location,
bool is_from_webstore,
bool is_hosted_app,
const std::string& app_url) {
int flags = is_from_webstore ?
extensions::Extension::FROM_WEBSTORE:
extensions::Extension::NO_FLAGS;
base::Value::Dict manifest;
manifest.Set("name", "NaCl Extension");
manifest.Set("version", "1");
manifest.Set("manifest_version", 2);
if (is_hosted_app) {
base::Value::List url_list;
url_list.Append(app_url);
manifest.SetByDottedPath(extensions::manifest_keys::kWebURLs,
std::move(url_list));
manifest.SetByDottedPath(extensions::manifest_keys::kLaunchWebURL, app_url);
}
std::string error;
return extensions::Extension::Create(base::FilePath(), location, manifest,
flags, &error);
}
scoped_refptr<const extensions::Extension> CreateExtension(
bool is_from_webstore) {
return CreateTestExtension(ManifestLocation::kInternal, is_from_webstore,
kNotHostedApp, std::string());
}
scoped_refptr<const extensions::Extension> CreateExtensionWithLocation(
ManifestLocation location,
bool is_from_webstore) {
return CreateTestExtension(
location, is_from_webstore, kNotHostedApp, std::string());
}
scoped_refptr<const extensions::Extension> CreateHostedApp(
bool is_from_webstore, const std::string& app_url) {
return CreateTestExtension(ManifestLocation::kInternal, is_from_webstore,
kHostedApp, app_url);
}
TEST_F(ChromeContentRendererClientTest, ExtensionsClientInitialized) {
auto* extensions_client = extensions::ExtensionsClient::Get();
ASSERT_TRUE(extensions_client);
// Ensure that the availability map is initialized correctly.
const auto& map =
extensions_client->GetFeatureDelegatedAvailabilityCheckMap();
EXPECT_EQ(5u, map.size());
for (const auto* feature :
extension_test_util::GetExpectedDelegatedFeaturesForTest()) {
EXPECT_EQ(1u, map.count(feature));
}
}
#endif // BUILDFLAG(ENABLE_EXTENSIONS)
TEST_F(ChromeContentRendererClientTest, NaClRestriction) {
// Unknown content types have no NaCl module.
{
WebPluginInfo info;
EXPECT_EQ(GURL(),
ChromeContentRendererClient::GetNaClContentHandlerURL(
"application/x-foo", info));
}
// Known content types have a NaCl module.
{
WebPluginInfo info;
AddContentTypeHandler(&info, "application/x-foo", "www.foo.com");
EXPECT_EQ(GURL("www.foo.com"),
ChromeContentRendererClient::GetNaClContentHandlerURL(
"application/x-foo", info));
}
#if BUILDFLAG(ENABLE_NACL)
// --enable-nacl allows all NaCl apps.
{
EXPECT_TRUE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL(), kNaClUnrestricted,
CreateExtension(kExtensionNotFromWebStore).get()));
}
// Unpacked extensions are allowed without --enable-nacl.
{
EXPECT_TRUE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL(kExtensionUrl), kNaClRestricted,
CreateExtensionWithLocation(ManifestLocation::kUnpacked,
kExtensionNotFromWebStore)
.get()));
}
// Component extensions are allowed without --enable-nacl.
{
EXPECT_TRUE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL(kExtensionUrl), kNaClRestricted,
CreateExtensionWithLocation(ManifestLocation::kComponent,
kExtensionNotFromWebStore)
.get()));
}
{
EXPECT_TRUE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL(kExtensionUrl), kNaClRestricted,
CreateExtensionWithLocation(ManifestLocation::kExternalComponent,
kExtensionNotFromWebStore)
.get()));
}
// Extensions that are force installed by policy are allowed without
// --enable-nacl.
{
EXPECT_TRUE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL(kExtensionUrl), kNaClRestricted,
CreateExtensionWithLocation(ManifestLocation::kExternalPolicy,
kExtensionNotFromWebStore)
.get()));
EXPECT_TRUE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL(kExtensionUrl), kNaClRestricted,
CreateExtensionWithLocation(ManifestLocation::kExternalPolicyDownload,
kExtensionNotFromWebStore)
.get()));
}
// CWS extensions are allowed without --enable-nacl if called from an
// extension url.
{
EXPECT_TRUE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL(kExtensionUrl), kNaClRestricted,
CreateExtension(kExtensionFromWebStore).get()));
}
// Other URLs (including previously-whitelisted URLs) are blocked
// without --enable-nacl.
{
EXPECT_FALSE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL("https://plus.google.com.evil.com/foo1"), kNaClRestricted,
nullptr));
EXPECT_FALSE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL("https://talkgadget.google.com/hangouts/foo1"), kNaClRestricted,
nullptr));
}
// Non chrome-extension:// URLs belonging to hosted apps are allowed for
// webstore installed hosted apps.
{
EXPECT_TRUE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL("http://example.com/test.html"), kNaClRestricted,
CreateHostedApp(kExtensionFromWebStore, "http://example.com/").get()));
EXPECT_FALSE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL("http://example.com/test.html"), kNaClRestricted,
CreateHostedApp(kExtensionNotFromWebStore, "http://example.com/")
.get()));
EXPECT_FALSE(ChromeContentRendererClient::IsNativeNaClAllowed(
GURL("http://example.evil.com/test.html"), kNaClRestricted,
CreateHostedApp(kExtensionNotFromWebStore, "http://example.com/")
.get()));
}
#endif // BUILDFLAG(ENABLE_NACL)
}