// 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 "extensions/renderer/module_system_test.h" #include #include #include "base/callback.h" #include "base/files/file_path.h" #include "base/files/file_util.h" #include "base/lazy_instance.h" #include "base/memory/scoped_ptr.h" #include "base/path_service.h" #include "base/stl_util.h" #include "base/strings/string_piece.h" #include "extensions/common/extension_paths.h" #include "extensions/renderer/logging_native_handler.h" #include "extensions/renderer/object_backed_native_handler.h" #include "extensions/renderer/safe_builtins.h" #include "extensions/renderer/utils_native_handler.h" #include "ui/base/resource/resource_bundle.h" namespace extensions { namespace { class FailsOnException : public ModuleSystem::ExceptionHandler { public: FailsOnException() : ModuleSystem::ExceptionHandler(nullptr) {} void HandleUncaughtException(const v8::TryCatch& try_catch) override { FAIL() << "Uncaught exception: " << CreateExceptionString(try_catch); } }; class V8ExtensionConfigurator { public: V8ExtensionConfigurator() : safe_builtins_(SafeBuiltins::CreateV8Extension()), names_(1, safe_builtins_->name()), configuration_( new v8::ExtensionConfiguration(static_cast(names_.size()), vector_as_array(&names_))) { v8::RegisterExtension(safe_builtins_.get()); } v8::ExtensionConfiguration* GetConfiguration() { return configuration_.get(); } private: scoped_ptr safe_builtins_; std::vector names_; scoped_ptr configuration_; }; base::LazyInstance::Leaky g_v8_extension_configurator = LAZY_INSTANCE_INITIALIZER; } // namespace // Native JS functions for doing asserts. class ModuleSystemTestEnvironment::AssertNatives : public ObjectBackedNativeHandler { public: explicit AssertNatives(ScriptContext* context) : ObjectBackedNativeHandler(context), assertion_made_(false), failed_(false) { RouteFunction( "AssertTrue", base::Bind(&AssertNatives::AssertTrue, base::Unretained(this))); RouteFunction( "AssertFalse", base::Bind(&AssertNatives::AssertFalse, base::Unretained(this))); } bool assertion_made() { return assertion_made_; } bool failed() { return failed_; } void AssertTrue(const v8::FunctionCallbackInfo& args) { CHECK_EQ(1, args.Length()); assertion_made_ = true; failed_ = failed_ || !args[0]->ToBoolean(args.GetIsolate())->Value(); } void AssertFalse(const v8::FunctionCallbackInfo& args) { CHECK_EQ(1, args.Length()); assertion_made_ = true; failed_ = failed_ || args[0]->ToBoolean(args.GetIsolate())->Value(); } private: bool assertion_made_; bool failed_; }; // Source map that operates on std::strings. class ModuleSystemTestEnvironment::StringSourceMap : public ModuleSystem::SourceMap { public: StringSourceMap() {} ~StringSourceMap() override {} v8::Local GetSource(v8::Isolate* isolate, const std::string& name) override { if (source_map_.count(name) == 0) return v8::Undefined(isolate); return v8::String::NewFromUtf8(isolate, source_map_[name].c_str()); } bool Contains(const std::string& name) override { return source_map_.count(name); } void RegisterModule(const std::string& name, const std::string& source) { CHECK_EQ(0u, source_map_.count(name)) << "Module " << name << " not found"; source_map_[name] = source; } private: std::map source_map_; }; ModuleSystemTestEnvironment::ModuleSystemTestEnvironment(v8::Isolate* isolate) : isolate_(isolate), context_holder_(new gin::ContextHolder(isolate_)), handle_scope_(isolate_), source_map_(new StringSourceMap()) { context_holder_->SetContext(v8::Context::New( isolate, g_v8_extension_configurator.Get().GetConfiguration())); context_.reset(new ScriptContext(context_holder_->context(), nullptr, // WebFrame nullptr, // Extension Feature::BLESSED_EXTENSION_CONTEXT, nullptr, // Effective Extension Feature::BLESSED_EXTENSION_CONTEXT)); context_->v8_context()->Enter(); assert_natives_ = new AssertNatives(context_.get()); { scoped_ptr module_system( new ModuleSystem(context_.get(), source_map_.get())); context_->set_module_system(module_system.Pass()); } ModuleSystem* module_system = context_->module_system(); module_system->RegisterNativeHandler( "assert", scoped_ptr(assert_natives_)); module_system->RegisterNativeHandler( "logging", scoped_ptr(new LoggingNativeHandler(context_.get()))); module_system->RegisterNativeHandler( "utils", scoped_ptr(new UtilsNativeHandler(context_.get()))); module_system->SetExceptionHandlerForTest( scoped_ptr(new FailsOnException)); } ModuleSystemTestEnvironment::~ModuleSystemTestEnvironment() { if (context_->is_valid()) ShutdownModuleSystem(); } void ModuleSystemTestEnvironment::RegisterModule(const std::string& name, const std::string& code) { source_map_->RegisterModule(name, code); } void ModuleSystemTestEnvironment::RegisterModule(const std::string& name, int resource_id) { const std::string& code = ResourceBundle::GetSharedInstance() .GetRawDataResource(resource_id) .as_string(); source_map_->RegisterModule(name, code); } void ModuleSystemTestEnvironment::OverrideNativeHandler( const std::string& name, const std::string& code) { RegisterModule(name, code); context_->module_system()->OverrideNativeHandlerForTest(name); } void ModuleSystemTestEnvironment::RegisterTestFile( const std::string& module_name, const std::string& file_name) { base::FilePath test_js_file_path; ASSERT_TRUE(PathService::Get(DIR_TEST_DATA, &test_js_file_path)); test_js_file_path = test_js_file_path.AppendASCII(file_name); std::string test_js; ASSERT_TRUE(base::ReadFileToString(test_js_file_path, &test_js)); source_map_->RegisterModule(module_name, test_js); } void ModuleSystemTestEnvironment::ShutdownGin() { context_holder_.reset(); } void ModuleSystemTestEnvironment::ShutdownModuleSystem() { CHECK(context_->is_valid()); context_->v8_context()->Exit(); context_->Invalidate(); } v8::Local ModuleSystemTestEnvironment::CreateGlobal( const std::string& name) { v8::EscapableHandleScope handle_scope(isolate_); v8::Local object = v8::Object::New(isolate_); isolate_->GetCurrentContext()->Global()->Set( v8::String::NewFromUtf8(isolate_, name.c_str()), object); return handle_scope.Escape(object); } ModuleSystemTest::ModuleSystemTest() : isolate_(v8::Isolate::GetCurrent()), should_assertions_be_made_(true) { } ModuleSystemTest::~ModuleSystemTest() { } void ModuleSystemTest::SetUp() { env_ = CreateEnvironment(); } void ModuleSystemTest::TearDown() { // All tests must assert at least once unless otherwise specified. EXPECT_EQ(should_assertions_be_made_, env_->assert_natives()->assertion_made()); EXPECT_FALSE(env_->assert_natives()->failed()); env_.reset(); v8::HeapStatistics stats; isolate_->GetHeapStatistics(&stats); size_t old_heap_size = 0; // Run the GC until the heap size reaches a steady state to ensure that // all the garbage is collected. while (stats.used_heap_size() != old_heap_size) { old_heap_size = stats.used_heap_size(); isolate_->RequestGarbageCollectionForTesting( v8::Isolate::kFullGarbageCollection); isolate_->GetHeapStatistics(&stats); } } scoped_ptr ModuleSystemTest::CreateEnvironment() { return make_scoped_ptr(new ModuleSystemTestEnvironment(isolate_)); } void ModuleSystemTest::ExpectNoAssertionsMade() { should_assertions_be_made_ = false; } void ModuleSystemTest::RunResolvedPromises() { isolate_->RunMicrotasks(); } } // namespace extensions