• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "extensions/common/user_script.h"
6 
7 #include "base/command_line.h"
8 #include "base/pickle.h"
9 #include "base/strings/string_util.h"
10 #include "extensions/common/switches.h"
11 
12 namespace {
13 
UrlMatchesGlobs(const std::vector<std::string> * globs,const GURL & url)14 bool UrlMatchesGlobs(const std::vector<std::string>* globs,
15                      const GURL& url) {
16   for (std::vector<std::string>::const_iterator glob = globs->begin();
17        glob != globs->end(); ++glob) {
18     if (MatchPattern(url.spec(), *glob))
19       return true;
20   }
21 
22   return false;
23 }
24 
25 }  // namespace
26 
27 namespace extensions {
28 
29 // The bitmask for valid user script injectable schemes used by URLPattern.
30 enum {
31   kValidUserScriptSchemes = URLPattern::SCHEME_CHROMEUI |
32                             URLPattern::SCHEME_HTTP |
33                             URLPattern::SCHEME_HTTPS |
34                             URLPattern::SCHEME_FILE |
35                             URLPattern::SCHEME_FTP
36 };
37 
38 // static
39 const char UserScript::kFileExtension[] = ".user.js";
40 
IsURLUserScript(const GURL & url,const std::string & mime_type)41 bool UserScript::IsURLUserScript(const GURL& url,
42                                  const std::string& mime_type) {
43   return EndsWith(url.ExtractFileName(), kFileExtension, false) &&
44       mime_type != "text/html";
45 }
46 
47 // static
ValidUserScriptSchemes(bool canExecuteScriptEverywhere)48 int UserScript::ValidUserScriptSchemes(bool canExecuteScriptEverywhere) {
49   if (canExecuteScriptEverywhere)
50     return URLPattern::SCHEME_ALL;
51   int valid_schemes = kValidUserScriptSchemes;
52   if (!CommandLine::ForCurrentProcess()->HasSwitch(
53       switches::kExtensionsOnChromeURLs)) {
54     valid_schemes &= ~URLPattern::SCHEME_CHROMEUI;
55   }
56   return valid_schemes;
57 }
58 
File(const base::FilePath & extension_root,const base::FilePath & relative_path,const GURL & url)59 UserScript::File::File(const base::FilePath& extension_root,
60                        const base::FilePath& relative_path,
61                        const GURL& url)
62     : extension_root_(extension_root),
63       relative_path_(relative_path),
64       url_(url) {
65 }
66 
File()67 UserScript::File::File() {}
68 
~File()69 UserScript::File::~File() {}
70 
UserScript()71 UserScript::UserScript()
72     : run_location_(DOCUMENT_IDLE),
73       user_script_id_(-1),
74       emulate_greasemonkey_(false),
75       match_all_frames_(false),
76       match_about_blank_(false),
77       incognito_enabled_(false) {}
78 
~UserScript()79 UserScript::~UserScript() {
80 }
81 
add_url_pattern(const URLPattern & pattern)82 void UserScript::add_url_pattern(const URLPattern& pattern) {
83   url_set_.AddPattern(pattern);
84 }
85 
add_exclude_url_pattern(const URLPattern & pattern)86 void UserScript::add_exclude_url_pattern(const URLPattern& pattern) {
87   exclude_url_set_.AddPattern(pattern);
88 }
89 
MatchesURL(const GURL & url) const90 bool UserScript::MatchesURL(const GURL& url) const {
91   if (!url_set_.is_empty()) {
92     if (!url_set_.MatchesURL(url))
93       return false;
94   }
95 
96   if (!exclude_url_set_.is_empty()) {
97     if (exclude_url_set_.MatchesURL(url))
98       return false;
99   }
100 
101   if (!globs_.empty()) {
102     if (!UrlMatchesGlobs(&globs_, url))
103       return false;
104   }
105 
106   if (!exclude_globs_.empty()) {
107     if (UrlMatchesGlobs(&exclude_globs_, url))
108       return false;
109   }
110 
111   return true;
112 }
113 
Pickle(::Pickle * pickle) const114 void UserScript::File::Pickle(::Pickle* pickle) const {
115   pickle->WriteString(url_.spec());
116   // Do not write path. It's not needed in the renderer.
117   // Do not write content. It will be serialized by other means.
118 }
119 
Unpickle(const::Pickle & pickle,PickleIterator * iter)120 void UserScript::File::Unpickle(const ::Pickle& pickle, PickleIterator* iter) {
121   // Read the url from the pickle.
122   std::string url;
123   CHECK(pickle.ReadString(iter, &url));
124   set_url(GURL(url));
125 }
126 
Pickle(::Pickle * pickle) const127 void UserScript::Pickle(::Pickle* pickle) const {
128   // Write the simple types to the pickle.
129   pickle->WriteInt(run_location());
130   pickle->WriteString(extension_id());
131   pickle->WriteInt64(user_script_id_);
132   pickle->WriteBool(emulate_greasemonkey());
133   pickle->WriteBool(match_all_frames());
134   pickle->WriteBool(match_about_blank());
135   pickle->WriteBool(is_incognito_enabled());
136 
137   PickleGlobs(pickle, globs_);
138   PickleGlobs(pickle, exclude_globs_);
139   PickleURLPatternSet(pickle, url_set_);
140   PickleURLPatternSet(pickle, exclude_url_set_);
141   PickleScripts(pickle, js_scripts_);
142   PickleScripts(pickle, css_scripts_);
143 }
144 
PickleGlobs(::Pickle * pickle,const std::vector<std::string> & globs) const145 void UserScript::PickleGlobs(::Pickle* pickle,
146                              const std::vector<std::string>& globs) const {
147   pickle->WriteUInt64(globs.size());
148   for (std::vector<std::string>::const_iterator glob = globs.begin();
149        glob != globs.end(); ++glob) {
150     pickle->WriteString(*glob);
151   }
152 }
153 
PickleURLPatternSet(::Pickle * pickle,const URLPatternSet & pattern_list) const154 void UserScript::PickleURLPatternSet(::Pickle* pickle,
155                                      const URLPatternSet& pattern_list) const {
156   pickle->WriteUInt64(pattern_list.patterns().size());
157   for (URLPatternSet::const_iterator pattern = pattern_list.begin();
158        pattern != pattern_list.end(); ++pattern) {
159     pickle->WriteInt(pattern->valid_schemes());
160     pickle->WriteString(pattern->GetAsString());
161   }
162 }
163 
PickleScripts(::Pickle * pickle,const FileList & scripts) const164 void UserScript::PickleScripts(::Pickle* pickle,
165                                const FileList& scripts) const {
166   pickle->WriteUInt64(scripts.size());
167   for (FileList::const_iterator file = scripts.begin();
168        file != scripts.end(); ++file) {
169     file->Pickle(pickle);
170   }
171 }
172 
Unpickle(const::Pickle & pickle,PickleIterator * iter)173 void UserScript::Unpickle(const ::Pickle& pickle, PickleIterator* iter) {
174   // Read the run location.
175   int run_location = 0;
176   CHECK(pickle.ReadInt(iter, &run_location));
177   CHECK(run_location >= 0 && run_location < RUN_LOCATION_LAST);
178   run_location_ = static_cast<RunLocation>(run_location);
179 
180   CHECK(pickle.ReadString(iter, &extension_id_));
181   CHECK(pickle.ReadInt64(iter, &user_script_id_));
182   CHECK(pickle.ReadBool(iter, &emulate_greasemonkey_));
183   CHECK(pickle.ReadBool(iter, &match_all_frames_));
184   CHECK(pickle.ReadBool(iter, &match_about_blank_));
185   CHECK(pickle.ReadBool(iter, &incognito_enabled_));
186 
187   UnpickleGlobs(pickle, iter, &globs_);
188   UnpickleGlobs(pickle, iter, &exclude_globs_);
189   UnpickleURLPatternSet(pickle, iter, &url_set_);
190   UnpickleURLPatternSet(pickle, iter, &exclude_url_set_);
191   UnpickleScripts(pickle, iter, &js_scripts_);
192   UnpickleScripts(pickle, iter, &css_scripts_);
193 }
194 
UnpickleGlobs(const::Pickle & pickle,PickleIterator * iter,std::vector<std::string> * globs)195 void UserScript::UnpickleGlobs(const ::Pickle& pickle, PickleIterator* iter,
196                                std::vector<std::string>* globs) {
197   uint64 num_globs = 0;
198   CHECK(pickle.ReadUInt64(iter, &num_globs));
199   globs->clear();
200   for (uint64 i = 0; i < num_globs; ++i) {
201     std::string glob;
202     CHECK(pickle.ReadString(iter, &glob));
203     globs->push_back(glob);
204   }
205 }
206 
UnpickleURLPatternSet(const::Pickle & pickle,PickleIterator * iter,URLPatternSet * pattern_list)207 void UserScript::UnpickleURLPatternSet(const ::Pickle& pickle,
208                                        PickleIterator* iter,
209                                        URLPatternSet* pattern_list) {
210   uint64 num_patterns = 0;
211   CHECK(pickle.ReadUInt64(iter, &num_patterns));
212 
213   pattern_list->ClearPatterns();
214   for (uint64 i = 0; i < num_patterns; ++i) {
215     int valid_schemes;
216     CHECK(pickle.ReadInt(iter, &valid_schemes));
217 
218     std::string pattern_str;
219     CHECK(pickle.ReadString(iter, &pattern_str));
220 
221     URLPattern pattern(kValidUserScriptSchemes);
222     URLPattern::ParseResult result = pattern.Parse(pattern_str);
223     CHECK(URLPattern::PARSE_SUCCESS == result) <<
224         URLPattern::GetParseResultString(result) << " " << pattern_str.c_str();
225 
226     pattern.SetValidSchemes(valid_schemes);
227     pattern_list->AddPattern(pattern);
228   }
229 }
230 
UnpickleScripts(const::Pickle & pickle,PickleIterator * iter,FileList * scripts)231 void UserScript::UnpickleScripts(const ::Pickle& pickle, PickleIterator* iter,
232                                  FileList* scripts) {
233   uint64 num_files = 0;
234   CHECK(pickle.ReadUInt64(iter, &num_files));
235   scripts->clear();
236   for (uint64 i = 0; i < num_files; ++i) {
237     File file;
238     file.Unpickle(pickle, iter);
239     scripts->push_back(file);
240   }
241 }
242 
243 }  // namespace extensions
244