// Copyright (c) 2011 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/common/extensions/user_script.h" #include "base/pickle.h" #include "base/string_util.h" namespace { bool UrlMatchesGlobs(const std::vector* globs, const GURL& url) { for (std::vector::const_iterator glob = globs->begin(); glob != globs->end(); ++glob) { if (MatchPattern(url.spec(), *glob)) return true; } return false; } } // namespace // static const char UserScript::kFileExtension[] = ".user.js"; // static const int UserScript::kValidUserScriptSchemes = URLPattern::SCHEME_HTTP | URLPattern::SCHEME_HTTPS | URLPattern::SCHEME_FILE | URLPattern::SCHEME_FTP; bool UserScript::IsURLUserScript(const GURL& url, const std::string& mime_type) { return EndsWith(url.ExtractFileName(), kFileExtension, false) && mime_type != "text/html"; } UserScript::File::File(const FilePath& extension_root, const FilePath& relative_path, const GURL& url) : extension_root_(extension_root), relative_path_(relative_path), url_(url) { } UserScript::File::File() {} UserScript::File::~File() {} UserScript::UserScript() : run_location_(DOCUMENT_IDLE), emulate_greasemonkey_(false), match_all_frames_(false), incognito_enabled_(false) { } UserScript::~UserScript() { } void UserScript::add_url_pattern(const URLPattern& pattern) { url_set_.AddPattern(pattern); } void UserScript::add_exclude_url_pattern(const URLPattern& pattern) { exclude_url_set_.AddPattern(pattern); } bool UserScript::MatchesURL(const GURL& url) const { if (!url_set_.is_empty()) { if (!url_set_.MatchesURL(url)) return false; } if (!exclude_url_set_.is_empty()) { if (exclude_url_set_.MatchesURL(url)) return false; } if (!globs_.empty()) { if (!UrlMatchesGlobs(&globs_, url)) return false; } if (!exclude_globs_.empty()) { if (UrlMatchesGlobs(&exclude_globs_, url)) return false; } return true; } void UserScript::File::Pickle(::Pickle* pickle) const { pickle->WriteString(url_.spec()); // Do not write path. It's not needed in the renderer. // Do not write content. It will be serialized by other means. } void UserScript::File::Unpickle(const ::Pickle& pickle, void** iter) { // Read url. std::string url; CHECK(pickle.ReadString(iter, &url)); set_url(GURL(url)); } void UserScript::Pickle(::Pickle* pickle) const { // Write simple types. pickle->WriteInt(run_location()); pickle->WriteString(extension_id()); pickle->WriteBool(emulate_greasemonkey()); pickle->WriteBool(match_all_frames()); pickle->WriteBool(is_incognito_enabled()); // Write globs. std::vector::const_iterator glob; pickle->WriteSize(globs_.size()); for (glob = globs_.begin(); glob != globs_.end(); ++glob) { pickle->WriteString(*glob); } pickle->WriteSize(exclude_globs_.size()); for (glob = exclude_globs_.begin(); glob != exclude_globs_.end(); ++glob) { pickle->WriteString(*glob); } // Write url patterns. URLPatternSet pattern_list = url_set_; pickle->WriteSize(pattern_list.patterns().size()); for (URLPatternSet::const_iterator pattern = pattern_list.begin(); pattern != pattern_list.end(); ++pattern) { pickle->WriteInt(pattern->valid_schemes()); pickle->WriteString(pattern->GetAsString()); } // Write js scripts. pickle->WriteSize(js_scripts_.size()); for (FileList::const_iterator file = js_scripts_.begin(); file != js_scripts_.end(); ++file) { file->Pickle(pickle); } // Write css scripts. pickle->WriteSize(css_scripts_.size()); for (FileList::const_iterator file = css_scripts_.begin(); file != css_scripts_.end(); ++file) { file->Pickle(pickle); } } void UserScript::Unpickle(const ::Pickle& pickle, void** iter) { // Read the run location. int run_location = 0; CHECK(pickle.ReadInt(iter, &run_location)); CHECK(run_location >= 0 && run_location < RUN_LOCATION_LAST); run_location_ = static_cast(run_location); CHECK(pickle.ReadString(iter, &extension_id_)); CHECK(pickle.ReadBool(iter, &emulate_greasemonkey_)); CHECK(pickle.ReadBool(iter, &match_all_frames_)); CHECK(pickle.ReadBool(iter, &incognito_enabled_)); // Read globs. size_t num_globs = 0; CHECK(pickle.ReadSize(iter, &num_globs)); globs_.clear(); for (size_t i = 0; i < num_globs; ++i) { std::string glob; CHECK(pickle.ReadString(iter, &glob)); globs_.push_back(glob); } CHECK(pickle.ReadSize(iter, &num_globs)); exclude_globs_.clear(); for (size_t i = 0; i < num_globs; ++i) { std::string glob; CHECK(pickle.ReadString(iter, &glob)); exclude_globs_.push_back(glob); } // Read url patterns. size_t num_patterns = 0; CHECK(pickle.ReadSize(iter, &num_patterns)); url_set_.ClearPatterns(); for (size_t i = 0; i < num_patterns; ++i) { int valid_schemes; CHECK(pickle.ReadInt(iter, &valid_schemes)); std::string pattern_str; URLPattern pattern(valid_schemes); CHECK(pickle.ReadString(iter, &pattern_str)); // We remove the file scheme if it's not actually allowed (see Extension:: // LoadUserScriptHelper), but we need it temporarily while loading the // pattern so that it's valid. bool had_file_scheme = (valid_schemes & URLPattern::SCHEME_FILE) != 0; if (!had_file_scheme) pattern.SetValidSchemes(valid_schemes | URLPattern::SCHEME_FILE); CHECK(URLPattern::PARSE_SUCCESS == pattern.Parse(pattern_str, URLPattern::IGNORE_PORTS)); if (!had_file_scheme) pattern.SetValidSchemes(valid_schemes); url_set_.AddPattern(pattern); } // Read js scripts. size_t num_js_files = 0; CHECK(pickle.ReadSize(iter, &num_js_files)); js_scripts_.clear(); for (size_t i = 0; i < num_js_files; ++i) { File file; file.Unpickle(pickle, iter); js_scripts_.push_back(file); } // Read css scripts. size_t num_css_files = 0; CHECK(pickle.ReadSize(iter, &num_css_files)); css_scripts_.clear(); for (size_t i = 0; i < num_css_files; ++i) { File file; file.Unpickle(pickle, iter); css_scripts_.push_back(file); } }