// Copyright (c) 2009 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 "chrome/renderer/user_script_slave.h" #include "app/resource_bundle.h" #include "base/command_line.h" #include "base/histogram.h" #include "base/logging.h" #include "base/perftimer.h" #include "base/pickle.h" #include "base/shared_memory.h" #include "base/string_util.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/extensions/extension.h" #include "chrome/common/extensions/extension_constants.h" #include "chrome/renderer/extension_groups.h" #include "chrome/renderer/render_thread.h" #include "googleurl/src/gurl.h" #include "third_party/WebKit/WebKit/chromium/public/WebFrame.h" #include "grit/renderer_resources.h" using WebKit::WebFrame; using WebKit::WebString; // These two strings are injected before and after the Greasemonkey API and // user script to wrap it in an anonymous scope. static const char kUserScriptHead[] = "(function (unsafeWindow) {\n"; static const char kUserScriptTail[] = "\n})(window);"; // Sets up the chrome.extension module. This may be run multiple times per // context, but the init method deletes itself after the first time. static const char kInitExtension[] = "if (chrome.initExtension) chrome.initExtension('%s', true, %s);"; int UserScriptSlave::GetIsolatedWorldId(const std::string& extension_id) { typedef std::map IsolatedWorldMap; static IsolatedWorldMap g_isolated_world_ids; static int g_next_isolated_world_id = 1; IsolatedWorldMap::iterator iter = g_isolated_world_ids.find(extension_id); if (iter != g_isolated_world_ids.end()) return iter->second; int new_id = g_next_isolated_world_id; ++g_next_isolated_world_id; // This map will tend to pile up over time, but realistically, you're never // going to have enough extensions for it to matter. g_isolated_world_ids[extension_id] = new_id; return new_id; } UserScriptSlave::UserScriptSlave() : shared_memory_(NULL), script_deleter_(&scripts_) { api_js_ = ResourceBundle::GetSharedInstance().GetRawDataResource( IDR_GREASEMONKEY_API_JS); } void UserScriptSlave::GetActiveExtensions(std::set* extension_ids) { for (size_t i = 0; i < scripts_.size(); ++i) { DCHECK(!scripts_[i]->extension_id().empty()); extension_ids->insert(scripts_[i]->extension_id()); } } bool UserScriptSlave::UpdateScripts(base::SharedMemoryHandle shared_memory) { scripts_.clear(); bool only_inject_incognito = RenderThread::current()->is_incognito_process(); // Create the shared memory object (read only). shared_memory_.reset(new base::SharedMemory(shared_memory, true)); if (!shared_memory_.get()) return false; // First get the size of the memory block. if (!shared_memory_->Map(sizeof(Pickle::Header))) return false; Pickle::Header* pickle_header = reinterpret_cast(shared_memory_->memory()); // Now map in the rest of the block. int pickle_size = sizeof(Pickle::Header) + pickle_header->payload_size; shared_memory_->Unmap(); if (!shared_memory_->Map(pickle_size)) return false; // Unpickle scripts. void* iter = NULL; size_t num_scripts = 0; Pickle pickle(reinterpret_cast(shared_memory_->memory()), pickle_size); pickle.ReadSize(&iter, &num_scripts); scripts_.reserve(num_scripts); for (size_t i = 0; i < num_scripts; ++i) { scripts_.push_back(new UserScript()); UserScript* script = scripts_.back(); script->Unpickle(pickle, &iter); // Note that this is a pointer into shared memory. We don't own it. It gets // cleared up when the last renderer or browser process drops their // reference to the shared memory. for (size_t j = 0; j < script->js_scripts().size(); ++j) { const char* body = NULL; int body_length = 0; CHECK(pickle.ReadData(&iter, &body, &body_length)); script->js_scripts()[j].set_external_content( base::StringPiece(body, body_length)); } for (size_t j = 0; j < script->css_scripts().size(); ++j) { const char* body = NULL; int body_length = 0; CHECK(pickle.ReadData(&iter, &body, &body_length)); script->css_scripts()[j].set_external_content( base::StringPiece(body, body_length)); } if (only_inject_incognito && !script->is_incognito_enabled()) { // This script shouldn't run in an incognito tab. delete script; scripts_.pop_back(); } } return true; } // static void UserScriptSlave::InsertInitExtensionCode( std::vector* sources, const std::string& extension_id) { DCHECK(sources); bool incognito = RenderThread::current()->is_incognito_process(); sources->insert(sources->begin(), WebScriptSource(WebString::fromUTF8( StringPrintf(kInitExtension, extension_id.c_str(), incognito ? "true" : "false")))); } bool UserScriptSlave::InjectScripts(WebFrame* frame, UserScript::RunLocation location) { GURL frame_url = GURL(frame->url()); // Don't bother if this is not a URL we inject script into. if (!URLPattern::IsValidScheme(frame_url.scheme())) return true; // Don't inject user scripts into the gallery itself. This prevents // a user script from removing the "report abuse" link, for example. if (frame_url.host() == GURL(extension_urls::kGalleryBrowsePrefix).host()) return true; PerfTimer timer; int num_css = 0; int num_scripts = 0; for (size_t i = 0; i < scripts_.size(); ++i) { std::vector sources; UserScript* script = scripts_[i]; if (frame->parent() && !script->match_all_frames()) continue; // Only match subframes if the script declared it wanted to. if (!script->MatchesUrl(frame->url())) continue; // This frame doesn't match the script url pattern, skip it. // CSS files are always injected on document start before js scripts. if (location == UserScript::DOCUMENT_START) { num_css += script->css_scripts().size(); for (size_t j = 0; j < script->css_scripts().size(); ++j) { PerfTimer insert_timer; UserScript::File& file = script->css_scripts()[j]; frame->insertStyleText( WebString::fromUTF8(file.GetContent().as_string()), WebString()); UMA_HISTOGRAM_TIMES("Extensions.InjectCssTime", insert_timer.Elapsed()); } } if (script->run_location() == location) { num_scripts += script->js_scripts().size(); for (size_t j = 0; j < script->js_scripts().size(); ++j) { UserScript::File &file = script->js_scripts()[j]; std::string content = file.GetContent().as_string(); // We add this dumb function wrapper for standalone user script to // emulate what Greasemonkey does. if (script->is_standalone() || script->emulate_greasemonkey()) { content.insert(0, kUserScriptHead); content += kUserScriptTail; } sources.push_back( WebScriptSource(WebString::fromUTF8(content), file.url())); } } if (!sources.empty()) { int isolated_world_id = 0; // Emulate Greasemonkey API for scripts that were converted to extensions // and "standalone" user scripts. if (script->is_standalone() || script->emulate_greasemonkey()) { sources.insert(sources.begin(), WebScriptSource(WebString::fromUTF8(api_js_.as_string()))); } // Setup chrome.self to contain an Extension object with the correct // ID. if (!script->extension_id().empty()) { InsertInitExtensionCode(&sources, script->extension_id()); isolated_world_id = GetIsolatedWorldId(script->extension_id()); } PerfTimer exec_timer; frame->executeScriptInIsolatedWorld( isolated_world_id, &sources.front(), sources.size(), EXTENSION_GROUP_CONTENT_SCRIPTS); UMA_HISTOGRAM_TIMES("Extensions.InjectScriptTime", exec_timer.Elapsed()); } } // Log debug info. if (location == UserScript::DOCUMENT_START) { UMA_HISTOGRAM_COUNTS_100("Extensions.InjectStart_CssCount", num_css); UMA_HISTOGRAM_COUNTS_100("Extensions.InjectStart_ScriptCount", num_scripts); if (num_css || num_scripts) UMA_HISTOGRAM_TIMES("Extensions.InjectStart_Time", timer.Elapsed()); } else if (location == UserScript::DOCUMENT_END) { UMA_HISTOGRAM_COUNTS_100("Extensions.InjectEnd_ScriptCount", num_scripts); if (num_scripts) UMA_HISTOGRAM_TIMES("Extensions.InjectEnd_Time", timer.Elapsed()); } else if (location == UserScript::DOCUMENT_IDLE) { UMA_HISTOGRAM_COUNTS_100("Extensions.InjectIdle_ScriptCount", num_scripts); if (num_scripts) UMA_HISTOGRAM_TIMES("Extensions.InjectIdle_Time", timer.Elapsed()); } else { NOTREACHED(); } LOG(INFO) << "Injected " << num_scripts << " scripts and " << num_css << " css files into " << frame->url().spec().data(); return true; }