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
|
// Copyright 2016 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <memory>
#include "base/dcheck_is_on.h"
#include "base/functional/bind.h"
#include "base/memory/scoped_refptr.h"
#include "build/build_config.h"
#include "build/chromeos_buildflags.h"
#include "chrome/browser/extensions/extension_service_test_base.h"
#include "chrome/test/base/testing_browser_process.h"
#include "extensions/browser/extension_function.h"
#include "extensions/browser/extension_registrar.h"
#include "extensions/browser/extension_registry.h"
#include "extensions/common/extension_builder.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace extensions {
namespace {
void SuccessCallback(bool* did_respond,
ExtensionFunction::ResponseType type,
base::Value::List results,
const std::string& error,
mojom::ExtraResponseDataPtr) {
EXPECT_EQ(ExtensionFunction::ResponseType::kSucceeded, type);
*did_respond = true;
}
void FailCallback(bool* did_respond,
ExtensionFunction::ResponseType type,
base::Value::List results,
const std::string& error,
mojom::ExtraResponseDataPtr) {
EXPECT_EQ(ExtensionFunction::ResponseType::kFailed, type);
*did_respond = true;
}
class ValidationFunction : public ExtensionFunction {
public:
explicit ValidationFunction(bool should_succeed)
: should_succeed_(should_succeed), did_respond_(false) {
set_response_callback(base::BindOnce(
(should_succeed ? &SuccessCallback : &FailCallback), &did_respond_));
}
ResponseAction Run() override {
EXPECT_TRUE(should_succeed_);
return RespondNow(NoArguments());
}
bool did_respond() { return did_respond_; }
private:
~ValidationFunction() override = default;
bool should_succeed_;
bool did_respond_;
};
} // namespace
using ChromeExtensionFunctionUnitTest = ExtensionServiceTestBase;
#if BUILDFLAG(IS_WIN) || BUILDFLAG(IS_CHROMEOS)
#define MAYBE_SimpleFunctionTest DISABLED_SimpleFunctionTest
#else
#define MAYBE_SimpleFunctionTest SimpleFunctionTest
#endif
TEST_F(ChromeExtensionFunctionUnitTest, MAYBE_SimpleFunctionTest) {
scoped_refptr<ValidationFunction> function(new ValidationFunction(true));
function->RunWithValidation().Execute();
EXPECT_TRUE(function->did_respond());
}
TEST_F(ChromeExtensionFunctionUnitTest, BrowserShutdownValidationFunctionTest) {
TestingBrowserProcess::GetGlobal()->SetShuttingDown(true);
scoped_refptr<ValidationFunction> function(new ValidationFunction(false));
function->RunWithValidation().Execute();
TestingBrowserProcess::GetGlobal()->SetShuttingDown(false);
EXPECT_TRUE(function->did_respond());
}
// Verifies that destroying the ExtensionFunction without responding is ok if
// the extension has been unloaded.
TEST_F(ChromeExtensionFunctionUnitTest, DestructionWithoutResponseOnUnload) {
InitializeEmptyExtensionService();
scoped_refptr<const Extension> extension = ExtensionBuilder("foo").Build();
registrar()->AddExtension(extension);
ASSERT_TRUE(registry()->enabled_extensions().Contains(extension->id()));
auto function = base::MakeRefCounted<ValidationFunction>(false);
function->set_extension(extension);
function->SetBrowserContextForTesting(browser_context());
registrar()->DisableExtension(extension->id(),
{disable_reason::DISABLE_USER_ACTION});
ASSERT_TRUE(registry()->disabled_extensions().Contains(extension->id()));
// Destroying the extension function without responding if the extension has
// been unloaded should not cause a crash.
function.reset();
}
#if DCHECK_IS_ON()
using ChromeExtensionFunctionDeathTest = ChromeExtensionFunctionUnitTest;
// Verify that destroying the extension function without responding causes a
// DCHECK failure.
#if BUILDFLAG(IS_WIN)
#define MAYBE_DestructionWithoutResponse DISABLED_DestructionWithoutResponse
#else
#define MAYBE_DestructionWithoutResponse DestructionWithoutResponse
#endif
TEST_F(ChromeExtensionFunctionDeathTest, MAYBE_DestructionWithoutResponse) {
ASSERT_DEATH(
{
InitializeEmptyExtensionService();
scoped_refptr<const Extension> extension =
ExtensionBuilder("foo").Build();
registrar()->AddExtension(extension);
ASSERT_TRUE(registry()->enabled_extensions().Contains(extension->id()));
auto function = base::MakeRefCounted<ValidationFunction>(false);
function->set_extension(extension);
function.reset();
},
"");
}
#endif // DCHECK_IS_ON()
} // namespace extensions
|