1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
|
// Copyright 2014 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/common/pepper_permission_util.h"
#include <memory>
#include <set>
#include <string>
#include <utility>
#include "components/crx_file/id_util.h"
#include "components/version_info/version_info.h"
#include "extensions/common/extension_builder.h"
#include "extensions/common/extension_set.h"
#include "extensions/common/features/feature_channel.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace extensions {
namespace {
// Return an extension with |id| which imports a module with the given
// |import_id|.
scoped_refptr<const Extension> CreateExtensionImportingModule(
const std::string& import_id,
const std::string& id) {
base::Value::Dict manifest =
base::Value::Dict()
.Set("name", "Has Dependent Modules")
.Set("version", "1.0")
.Set("manifest_version", 2)
.Set("import", base::Value::List().Append(
base::Value::Dict().Set("id", import_id)));
return ExtensionBuilder()
.SetManifest(std::move(manifest))
.AddFlags(Extension::FROM_WEBSTORE)
.SetID(id)
.Build();
}
} // namespace
TEST(PepperPermissionUtilTest, ExtensionAllowed) {
ScopedCurrentChannel current_channel(version_info::Channel::UNKNOWN);
ExtensionSet extensions;
std::string allowed_id = crx_file::id_util::GenerateId("allowed_extension");
base::Value::Dict manifest = base::Value::Dict()
.Set("name", "Allowed Extension")
.Set("version", "1.0")
.Set("manifest_version", 2);
scoped_refptr<const Extension> ext = ExtensionBuilder()
.SetManifest(std::move(manifest))
.SetID(allowed_id)
.Build();
extensions.Insert(ext);
std::set<std::string> allowlist;
std::string url = std::string("chrome-extension://") + allowed_id +
std::string("/manifest.nmf");
std::string bad_scheme_url =
std::string("http://") + allowed_id + std::string("/manifest.nmf");
std::string bad_host_url = std::string("chrome-extension://") +
crx_file::id_util::GenerateId("bad_host");
EXPECT_FALSE(
IsExtensionOrSharedModuleAllowed(GURL(url), &extensions, allowlist));
allowlist.insert(allowed_id);
EXPECT_TRUE(
IsExtensionOrSharedModuleAllowed(GURL(url), &extensions, allowlist));
EXPECT_FALSE(IsExtensionOrSharedModuleAllowed(GURL(bad_scheme_url),
&extensions, allowlist));
EXPECT_FALSE(IsExtensionOrSharedModuleAllowed(GURL(bad_host_url), &extensions,
allowlist));
}
TEST(PepperPermissionUtilTest, SharedModuleAllowed) {
ScopedCurrentChannel current_channel(version_info::Channel::UNKNOWN);
ExtensionSet extensions;
std::string allowed_id = crx_file::id_util::GenerateId("extension_id");
std::string bad_id = crx_file::id_util::GenerateId("bad_id");
base::Value::Dict shared_module_manifest =
base::Value::Dict()
.Set("name", "Allowed Shared Module")
.Set("version", "1.0")
.Set("manifest_version", 2)
.Set("export",
base::Value::Dict()
.Set("resources", base::Value::List().Append("*"))
// Add the extension to the allowlist. This
// restricts import to |allowed_id| only.
.Set("whitelist", base::Value::List().Append( // nocheck
allowed_id)));
scoped_refptr<const Extension> shared_module =
ExtensionBuilder().SetManifest(std::move(shared_module_manifest)).Build();
scoped_refptr<const Extension> ext =
CreateExtensionImportingModule(shared_module->id(), allowed_id);
std::string extension_url =
std::string("chrome-extension://") + ext->id() + std::string("/foo.html");
std::set<std::string> allowlist;
// Important: allow *only* the shared module.
allowlist.insert(shared_module->id());
extensions.Insert(ext);
// This should fail because shared_module is not in the set of extensions.
EXPECT_FALSE(IsExtensionOrSharedModuleAllowed(GURL(extension_url),
&extensions, allowlist));
extensions.Insert(shared_module);
EXPECT_TRUE(IsExtensionOrSharedModuleAllowed(GURL(extension_url), &extensions,
allowlist));
scoped_refptr<const Extension> not_in_sm_allowlist =
CreateExtensionImportingModule(shared_module->id(), bad_id);
std::string not_in_sm_allowlist_url = std::string("chrome-extension://") +
not_in_sm_allowlist->id() +
std::string("/foo.html");
extensions.Insert(not_in_sm_allowlist);
// This should succeed, even though |not_in_sm_allowlist| is not allowed
// to use shared_module, because the pepper permission utility does not care
// about that allowlist. It is possible to install against the allowlist as
// an unpacked extension.
EXPECT_TRUE(IsExtensionOrSharedModuleAllowed(GURL(not_in_sm_allowlist_url),
&extensions, allowlist));
// Note that the allowlist should be empty after this call, so tests checking
// for failure to import will fail because of this.
allowlist.erase(shared_module->id());
EXPECT_FALSE(IsExtensionOrSharedModuleAllowed(GURL(extension_url),
&extensions, allowlist));
}
} // namespace extensions
|