blob: 8fc7d7158a3880fedbe637c8783805ae7f85b03d [file] [log] [blame]
// 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 "extensions/common/features/complex_feature.h"
#include <string>
#include <utility>
#include "base/test/bind.h"
#include "content/public/common/content_features.h"
#include "extensions/common/features/feature.h"
#include "extensions/common/features/simple_feature.h"
#include "extensions/common/manifest.h"
#include "extensions/common/mojom/context_type.mojom.h"
#include "extensions/test/test_context_data.h"
#include "testing/gtest/include/gtest/gtest.h"
using extensions::mojom::ManifestLocation;
namespace extensions {
TEST(ComplexFeatureTest, MultipleRulesAllowlist) {
const HashedExtensionId kIdFoo("fooabbbbccccddddeeeeffffgggghhhh");
const HashedExtensionId kIdBar("barabbbbccccddddeeeeffffgggghhhh");
std::vector<Feature*> features;
{
// Rule: "extension", allowlist "foo".
std::unique_ptr<SimpleFeature> simple_feature(new SimpleFeature());
simple_feature->set_allowlist({kIdFoo.value().c_str()});
simple_feature->set_extension_types({Manifest::TYPE_EXTENSION});
features.push_back(simple_feature.release());
}
{
// Rule: "legacy_packaged_app", allowlist "bar".
std::unique_ptr<SimpleFeature> simple_feature(new SimpleFeature());
simple_feature->set_allowlist({kIdBar.value().c_str()});
simple_feature->set_extension_types({Manifest::TYPE_LEGACY_PACKAGED_APP});
features.push_back(simple_feature.release());
}
std::unique_ptr<ComplexFeature> feature(new ComplexFeature(&features));
// Test match 1st rule.
EXPECT_EQ(Feature::IS_AVAILABLE,
feature
->IsAvailableToManifest(kIdFoo, Manifest::TYPE_EXTENSION,
ManifestLocation::kInvalidLocation,
Feature::UNSPECIFIED_PLATFORM,
Feature::GetCurrentPlatform(),
kUnspecifiedContextId)
.result());
// Test match 2nd rule.
EXPECT_EQ(
Feature::IS_AVAILABLE,
feature
->IsAvailableToManifest(
kIdBar, Manifest::TYPE_LEGACY_PACKAGED_APP,
ManifestLocation::kInvalidLocation, Feature::UNSPECIFIED_PLATFORM,
Feature::GetCurrentPlatform(), kUnspecifiedContextId)
.result());
// Test allowlist with wrong extension type.
EXPECT_NE(Feature::IS_AVAILABLE,
feature
->IsAvailableToManifest(kIdBar, Manifest::TYPE_EXTENSION,
ManifestLocation::kInvalidLocation,
Feature::UNSPECIFIED_PLATFORM,
Feature::GetCurrentPlatform(),
kUnspecifiedContextId)
.result());
EXPECT_NE(
Feature::IS_AVAILABLE,
feature
->IsAvailableToManifest(
kIdFoo, Manifest::TYPE_LEGACY_PACKAGED_APP,
ManifestLocation::kInvalidLocation, Feature::UNSPECIFIED_PLATFORM,
Feature::GetCurrentPlatform(), kUnspecifiedContextId)
.result());
}
// Tests that dependencies are correctly checked.
TEST(ComplexFeatureTest, Dependencies) {
std::vector<Feature*> features;
{
// Rule which depends on an extension-only feature
// (content_security_policy).
std::unique_ptr<SimpleFeature> simple_feature(new SimpleFeature());
simple_feature->set_dependencies({"manifest:content_security_policy"});
features.push_back(simple_feature.release());
}
{
// Rule which depends on an platform-app-only feature (videoCapture).
std::unique_ptr<SimpleFeature> simple_feature(new SimpleFeature());
simple_feature->set_dependencies({"permission:videoCapture"});
features.push_back(simple_feature.release());
}
std::unique_ptr<ComplexFeature> feature(new ComplexFeature(&features));
// Available to extensions because of the content_security_policy rule.
EXPECT_EQ(
Feature::IS_AVAILABLE,
feature
->IsAvailableToManifest(
HashedExtensionId(std::string(32, 'a')), Manifest::TYPE_EXTENSION,
ManifestLocation::kInvalidLocation, Feature::UNSPECIFIED_PLATFORM,
Feature::GetCurrentPlatform(), kUnspecifiedContextId)
.result());
// Available to platform apps because of the videoCapture rule.
EXPECT_EQ(Feature::IS_AVAILABLE,
feature
->IsAvailableToManifest(HashedExtensionId(std::string(32, 'b')),
Manifest::TYPE_PLATFORM_APP,
ManifestLocation::kInvalidLocation,
Feature::UNSPECIFIED_PLATFORM,
Feature::GetCurrentPlatform(),
kUnspecifiedContextId)
.result());
// Not available to hosted apps.
EXPECT_EQ(Feature::INVALID_TYPE,
feature
->IsAvailableToManifest(HashedExtensionId(std::string(32, 'c')),
Manifest::TYPE_HOSTED_APP,
ManifestLocation::kInvalidLocation,
Feature::UNSPECIFIED_PLATFORM,
Feature::GetCurrentPlatform(),
kUnspecifiedContextId)
.result());
}
TEST(ComplexFeatureTest, RequiresDelegatedAvailabilityCheck) {
std::vector<Feature*> features;
// Test a complex feature where |requires_delegated_availability_check| hasn't
// been set on any of its simple features.
{
{
// Feature which doesn't set |requires_delegated_availability_check|.
auto simple_feature = std::make_unique<SimpleFeature>();
features.push_back(simple_feature.release());
}
{
// Feature which doesn't set |requires_delegated_availability_check|.
auto simple_feature = std::make_unique<SimpleFeature>();
features.push_back(simple_feature.release());
}
ComplexFeature complex_feature(&features);
EXPECT_FALSE(complex_feature.RequiresDelegatedAvailabilityCheck());
EXPECT_FALSE(complex_feature.HasDelegatedAvailabilityCheckHandler());
}
uint32_t delegated_availability_check_call_count = 0;
uint32_t success_call_count = 2;
auto delegated_availability_check =
[&](const std::string& api_full_name, const Extension* extension,
mojom::ContextType context, const GURL& url,
Feature::Platform platform, int context_id, bool check_developer_mode,
const ContextData& context_data) {
++delegated_availability_check_call_count;
return delegated_availability_check_call_count == success_call_count;
};
// Test a complex feature where |requires_delegated_availability_check| is set
// on multiple sub-features. The first sub-feature that requires the
// availability check should fail, while the second sub-feature should pass.
// In this case, the delegated availability check handler should be called
// twice.
{
{
// Feature which doesn't set |requires_delegated_availability_check|.
auto simple_feature = std::make_unique<SimpleFeature>();
simple_feature->set_contexts({mojom::ContextType::kPrivilegedExtension});
features.push_back(simple_feature.release());
}
// Two features which set |requires_delegated_availability_check| to true.
{
auto simple_feature = std::make_unique<SimpleFeature>();
simple_feature->set_requires_delegated_availability_check(true);
features.push_back(simple_feature.release());
}
{
auto simple_feature = std::make_unique<SimpleFeature>();
simple_feature->set_requires_delegated_availability_check(true);
features.push_back(simple_feature.release());
}
ComplexFeature complex_feature(&features);
EXPECT_TRUE(complex_feature.RequiresDelegatedAvailabilityCheck());
EXPECT_FALSE(complex_feature.HasDelegatedAvailabilityCheckHandler());
// A call to SetDelegatedAvailabilityCheckHandler() should set the
// handler to the sub-features that require it.
complex_feature.SetDelegatedAvailabilityCheckHandler(
base::BindLambdaForTesting(delegated_availability_check));
EXPECT_TRUE(complex_feature.HasDelegatedAvailabilityCheckHandler());
// This feature should be available the second time that the delegated
// availability check is called.
EXPECT_EQ(Feature::IS_AVAILABLE,
complex_feature
.IsAvailableToContext(
/*extension=*/nullptr, mojom::ContextType::kUnspecified,
GURL(), kUnspecifiedContextId, TestContextData())
.result());
EXPECT_EQ(2u, delegated_availability_check_call_count);
}
}
} // namespace extensions