• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 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 "chrome/browser/extensions/activity_log/activity_actions.h"
6 
7 #include <string>
8 
9 #include "base/command_line.h"
10 #include "base/json/json_string_value_serializer.h"
11 #include "base/logging.h"
12 #include "base/memory/singleton.h"
13 #include "base/strings/string_number_conversions.h"
14 #include "base/strings/string_util.h"
15 #include "base/strings/stringprintf.h"
16 #include "chrome/browser/extensions/activity_log/activity_action_constants.h"
17 #include "chrome/browser/extensions/activity_log/fullstream_ui_policy.h"
18 #include "chrome/browser/ui/browser.h"
19 #include "chrome/common/chrome_switches.h"
20 #include "chrome/common/extensions/dom_action_types.h"
21 #include "content/public/browser/browser_thread.h"
22 #include "content/public/browser/web_contents.h"
23 #include "sql/statement.h"
24 
25 namespace constants = activity_log_constants;
26 
27 namespace {
28 
Serialize(const base::Value * value)29 std::string Serialize(const base::Value* value) {
30   std::string value_as_text;
31   if (!value) {
32     value_as_text = "null";
33   } else {
34     JSONStringValueSerializer serializer(&value_as_text);
35     serializer.SerializeAndOmitBinaryValues(*value);
36   }
37   return value_as_text;
38 }
39 
40 }  // namespace
41 
42 namespace extensions {
43 
44 using api::activity_log_private::ExtensionActivity;
45 
Action(const std::string & extension_id,const base::Time & time,const ActionType action_type,const std::string & api_name)46 Action::Action(const std::string& extension_id,
47                const base::Time& time,
48                const ActionType action_type,
49                const std::string& api_name)
50     : extension_id_(extension_id),
51       time_(time),
52       action_type_(action_type),
53       api_name_(api_name),
54       page_incognito_(false),
55       arg_incognito_(false),
56       count_(0) {}
57 
~Action()58 Action::~Action() {}
59 
60 // TODO(mvrable): As an optimization, we might return this directly if the
61 // refcount is one.  However, there are likely to be other stray references in
62 // many cases that will prevent this optimization.
Clone() const63 scoped_refptr<Action> Action::Clone() const {
64   scoped_refptr<Action> clone(
65       new Action(extension_id(), time(), action_type(), api_name()));
66   if (args())
67     clone->set_args(make_scoped_ptr(args()->DeepCopy()));
68   clone->set_page_url(page_url());
69   clone->set_page_title(page_title());
70   clone->set_page_incognito(page_incognito());
71   clone->set_arg_url(arg_url());
72   clone->set_arg_incognito(arg_incognito());
73   if (other())
74     clone->set_other(make_scoped_ptr(other()->DeepCopy()));
75   return clone;
76 }
77 
set_args(scoped_ptr<ListValue> args)78 void Action::set_args(scoped_ptr<ListValue> args) {
79   args_.reset(args.release());
80 }
81 
mutable_args()82 ListValue* Action::mutable_args() {
83   if (!args_.get()) {
84     args_.reset(new ListValue());
85   }
86   return args_.get();
87 }
88 
set_page_url(const GURL & page_url)89 void Action::set_page_url(const GURL& page_url) {
90   page_url_ = page_url;
91 }
92 
set_arg_url(const GURL & arg_url)93 void Action::set_arg_url(const GURL& arg_url) {
94   arg_url_ = arg_url;
95 }
96 
set_other(scoped_ptr<DictionaryValue> other)97 void Action::set_other(scoped_ptr<DictionaryValue> other) {
98   other_.reset(other.release());
99 }
100 
mutable_other()101 DictionaryValue* Action::mutable_other() {
102   if (!other_.get()) {
103     other_.reset(new DictionaryValue());
104   }
105   return other_.get();
106 }
107 
SerializePageUrl() const108 std::string Action::SerializePageUrl() const {
109   return (page_incognito() ? constants::kIncognitoUrl : "") + page_url().spec();
110 }
111 
ParsePageUrl(const std::string & url)112 void Action::ParsePageUrl(const std::string& url) {
113   set_page_incognito(StartsWithASCII(url, constants::kIncognitoUrl, true));
114   if (page_incognito())
115     set_page_url(GURL(url.substr(strlen(constants::kIncognitoUrl))));
116   else
117     set_page_url(GURL(url));
118 }
119 
SerializeArgUrl() const120 std::string Action::SerializeArgUrl() const {
121   return (arg_incognito() ? constants::kIncognitoUrl : "") + arg_url().spec();
122 }
123 
ParseArgUrl(const std::string & url)124 void Action::ParseArgUrl(const std::string& url) {
125   set_arg_incognito(StartsWithASCII(url, constants::kIncognitoUrl, true));
126   if (arg_incognito())
127     set_arg_url(GURL(url.substr(strlen(constants::kIncognitoUrl))));
128   else
129     set_arg_url(GURL(url));
130 }
131 
ConvertToExtensionActivity()132 scoped_ptr<ExtensionActivity> Action::ConvertToExtensionActivity() {
133   scoped_ptr<ExtensionActivity> result(new ExtensionActivity);
134 
135   // We do this translation instead of using the same enum because the database
136   // values need to be stable; this allows us to change the extension API
137   // without affecting the database.
138   switch (action_type()) {
139     case ACTION_API_CALL:
140       result->activity_type = ExtensionActivity::ACTIVITY_TYPE_API_CALL;
141       break;
142     case ACTION_API_EVENT:
143       result->activity_type = ExtensionActivity::ACTIVITY_TYPE_API_EVENT;
144       break;
145     case ACTION_CONTENT_SCRIPT:
146       result->activity_type = ExtensionActivity::ACTIVITY_TYPE_CONTENT_SCRIPT;
147       break;
148     case ACTION_DOM_ACCESS:
149       result->activity_type = ExtensionActivity::ACTIVITY_TYPE_DOM_ACCESS;
150       break;
151     case ACTION_DOM_EVENT:
152       result->activity_type = ExtensionActivity::ACTIVITY_TYPE_DOM_EVENT;
153       break;
154     case ACTION_WEB_REQUEST:
155       result->activity_type = ExtensionActivity::ACTIVITY_TYPE_WEB_REQUEST;
156       break;
157     case UNUSED_ACTION_API_BLOCKED:
158     case ACTION_ANY:
159     default:
160       // This shouldn't be reached, but some people might have old or otherwise
161       // weird db entries. Treat it like an API call if that happens.
162       result->activity_type = ExtensionActivity::ACTIVITY_TYPE_API_CALL;
163       break;
164   }
165 
166   result->extension_id.reset(new std::string(extension_id()));
167   result->time.reset(new double(time().ToJsTime()));
168   result->count.reset(new double(count()));
169   result->api_call.reset(new std::string(api_name()));
170   result->args.reset(new std::string(Serialize(args())));
171   if (page_url().is_valid()) {
172     if (!page_title().empty())
173       result->page_title.reset(new std::string(page_title()));
174     result->page_url.reset(new std::string(SerializePageUrl()));
175   }
176   if (arg_url().is_valid())
177     result->arg_url.reset(new std::string(SerializeArgUrl()));
178 
179   if (other()) {
180     scoped_ptr<ExtensionActivity::Other> other_field(
181         new ExtensionActivity::Other);
182     bool prerender;
183     if (other()->GetBooleanWithoutPathExpansion(constants::kActionPrerender,
184                                                 &prerender)) {
185       other_field->prerender.reset(new bool(prerender));
186     }
187     const DictionaryValue* web_request;
188     if (other()->GetDictionaryWithoutPathExpansion(constants::kActionWebRequest,
189                                                    &web_request)) {
190       other_field->web_request.reset(new std::string(
191           ActivityLogPolicy::Util::Serialize(web_request)));
192     }
193     std::string extra;
194     if (other()->GetStringWithoutPathExpansion(constants::kActionExtra, &extra))
195       other_field->extra.reset(new std::string(extra));
196     int dom_verb;
197     if (other()->GetIntegerWithoutPathExpansion(constants::kActionDomVerb,
198                                                 &dom_verb)) {
199       switch (static_cast<DomActionType::Type>(dom_verb)) {
200         case DomActionType::GETTER:
201           other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_GETTER;
202           break;
203         case DomActionType::SETTER:
204           other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_SETTER;
205           break;
206         case DomActionType::METHOD:
207           other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_METHOD;
208           break;
209         case DomActionType::INSERTED:
210           other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_INSERTED;
211           break;
212         case DomActionType::XHR:
213           other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_XHR;
214           break;
215         case DomActionType::WEBREQUEST:
216           other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_WEBREQUEST;
217           break;
218         case DomActionType::MODIFIED:
219           other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_MODIFIED;
220           break;
221         default:
222           other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_NONE;
223       }
224     } else {
225       other_field->dom_verb = ExtensionActivity::Other::DOM_VERB_NONE;
226     }
227     result->other.reset(other_field.release());
228   }
229 
230   return result.Pass();
231 }
232 
PrintForDebug() const233 std::string Action::PrintForDebug() const {
234   std::string result = "ID=" + extension_id() + " CATEGORY=";
235   switch (action_type_) {
236     case ACTION_API_CALL:
237       result += "api_call";
238       break;
239     case ACTION_API_EVENT:
240       result += "api_event_callback";
241       break;
242     case ACTION_WEB_REQUEST:
243       result += "webrequest";
244       break;
245     case ACTION_CONTENT_SCRIPT:
246       result += "content_script";
247       break;
248     case UNUSED_ACTION_API_BLOCKED:
249       // This is deprecated.
250       result += "api_blocked";
251       break;
252     case ACTION_DOM_EVENT:
253       result += "dom_event";
254       break;
255     case ACTION_DOM_ACCESS:
256       result += "dom_access";
257       break;
258     default:
259       result += base::StringPrintf("type%d", static_cast<int>(action_type_));
260   }
261 
262   result += " API=" + api_name_;
263   if (args_.get()) {
264     result += " ARGS=" + Serialize(args_.get());
265   }
266   if (page_url_.is_valid()) {
267     if (page_incognito_)
268       result += " PAGE_URL=(incognito)" + page_url_.spec();
269     else
270       result += " PAGE_URL=" + page_url_.spec();
271   }
272   if (!page_title_.empty()) {
273     StringValue title(page_title_);
274     result += " PAGE_TITLE=" + Serialize(&title);
275   }
276   if (arg_url_.is_valid()) {
277     if (arg_incognito_)
278       result += " ARG_URL=(incognito)" + arg_url_.spec();
279     else
280       result += " ARG_URL=" + arg_url_.spec();
281   }
282   if (other_.get()) {
283     result += " OTHER=" + Serialize(other_.get());
284   }
285 
286   result += base::StringPrintf(" COUNT=%d", count_);
287   return result;
288 }
289 
operator ()(const scoped_refptr<Action> & lhs,const scoped_refptr<Action> & rhs) const290 bool ActionComparator::operator()(
291     const scoped_refptr<Action>& lhs,
292     const scoped_refptr<Action>& rhs) const {
293   if (lhs->time() != rhs->time())
294     return lhs->time() < rhs->time();
295   else
296     return ActionComparatorExcludingTime()(lhs, rhs);
297 }
298 
operator ()(const scoped_refptr<Action> & lhs,const scoped_refptr<Action> & rhs) const299 bool ActionComparatorExcludingTime::operator()(
300     const scoped_refptr<Action>& lhs,
301     const scoped_refptr<Action>& rhs) const {
302   if (lhs->extension_id() != rhs->extension_id())
303     return lhs->extension_id() < rhs->extension_id();
304   if (lhs->action_type() != rhs->action_type())
305     return lhs->action_type() < rhs->action_type();
306   if (lhs->api_name() != rhs->api_name())
307     return lhs->api_name() < rhs->api_name();
308 
309   // args might be null; treat a null value as less than all non-null values,
310   // including the empty string.
311   if (!lhs->args() && rhs->args())
312     return true;
313   if (lhs->args() && !rhs->args())
314     return false;
315   if (lhs->args() && rhs->args()) {
316     std::string lhs_args = ActivityLogPolicy::Util::Serialize(lhs->args());
317     std::string rhs_args = ActivityLogPolicy::Util::Serialize(rhs->args());
318     if (lhs_args != rhs_args)
319       return lhs_args < rhs_args;
320   }
321 
322   // Compare URLs as strings, and treat the incognito flag as a separate field.
323   if (lhs->page_url().spec() != rhs->page_url().spec())
324     return lhs->page_url().spec() < rhs->page_url().spec();
325   if (lhs->page_incognito() != rhs->page_incognito())
326     return lhs->page_incognito() < rhs->page_incognito();
327 
328   if (lhs->page_title() != rhs->page_title())
329     return lhs->page_title() < rhs->page_title();
330 
331   if (lhs->arg_url().spec() != rhs->arg_url().spec())
332     return lhs->arg_url().spec() < rhs->arg_url().spec();
333   if (lhs->arg_incognito() != rhs->arg_incognito())
334     return lhs->arg_incognito() < rhs->arg_incognito();
335 
336   // other is treated much like the args field.
337   if (!lhs->other() && rhs->other())
338     return true;
339   if (lhs->other() && !rhs->other())
340     return false;
341   if (lhs->other() && rhs->other()) {
342     std::string lhs_other = ActivityLogPolicy::Util::Serialize(lhs->other());
343     std::string rhs_other = ActivityLogPolicy::Util::Serialize(rhs->other());
344     if (lhs_other != rhs_other)
345       return lhs_other < rhs_other;
346   }
347 
348   // All fields compare as equal if this point is reached.
349   return false;
350 }
351 
352 }  // namespace extensions
353