Home | History | Annotate | Download | only in common
      1 // Copyright (c) 2012 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 <string>
      6 
      7 #include "extensions/common/event_filter.h"
      8 
      9 #include "components/url_matcher/url_matcher_factory.h"
     10 #include "ipc/ipc_message.h"
     11 
     12 using url_matcher::URLMatcher;
     13 using url_matcher::URLMatcherConditionSet;
     14 using url_matcher::URLMatcherFactory;
     15 
     16 namespace extensions {
     17 
     18 EventFilter::EventMatcherEntry::EventMatcherEntry(
     19     scoped_ptr<EventMatcher> event_matcher,
     20     URLMatcher* url_matcher,
     21     const URLMatcherConditionSet::Vector& condition_sets)
     22     : event_matcher_(event_matcher.Pass()),
     23       url_matcher_(url_matcher) {
     24   for (URLMatcherConditionSet::Vector::const_iterator it =
     25        condition_sets.begin(); it != condition_sets.end(); it++)
     26     condition_set_ids_.push_back((*it)->id());
     27   url_matcher_->AddConditionSets(condition_sets);
     28 }
     29 
     30 EventFilter::EventMatcherEntry::~EventMatcherEntry() {
     31   url_matcher_->RemoveConditionSets(condition_set_ids_);
     32 }
     33 
     34 void EventFilter::EventMatcherEntry::DontRemoveConditionSetsInDestructor() {
     35   condition_set_ids_.clear();
     36 }
     37 
     38 EventFilter::EventFilter()
     39     : next_id_(0),
     40       next_condition_set_id_(0) {
     41 }
     42 
     43 EventFilter::~EventFilter() {
     44   // Normally when an event matcher entry is removed from event_matchers_ it
     45   // will remove its condition sets from url_matcher_, but as url_matcher_ is
     46   // being destroyed anyway there is no need to do that step here.
     47   for (EventMatcherMultiMap::iterator it = event_matchers_.begin();
     48        it != event_matchers_.end(); it++) {
     49     for (EventMatcherMap::iterator it2 = it->second.begin();
     50          it2 != it->second.end(); it2++) {
     51       it2->second->DontRemoveConditionSetsInDestructor();
     52     }
     53   }
     54 }
     55 
     56 EventFilter::MatcherID
     57 EventFilter::AddEventMatcher(const std::string& event_name,
     58                              scoped_ptr<EventMatcher> matcher) {
     59   MatcherID id = next_id_++;
     60   URLMatcherConditionSet::Vector condition_sets;
     61   if (!CreateConditionSets(id, matcher.get(), &condition_sets))
     62     return -1;
     63 
     64   for (URLMatcherConditionSet::Vector::iterator it = condition_sets.begin();
     65        it != condition_sets.end(); it++) {
     66     condition_set_id_to_event_matcher_id_.insert(
     67         std::make_pair((*it)->id(), id));
     68   }
     69   id_to_event_name_[id] = event_name;
     70   event_matchers_[event_name][id] = linked_ptr<EventMatcherEntry>(
     71       new EventMatcherEntry(matcher.Pass(), &url_matcher_, condition_sets));
     72   return id;
     73 }
     74 
     75 EventMatcher* EventFilter::GetEventMatcher(MatcherID id) {
     76   DCHECK(id_to_event_name_.find(id) != id_to_event_name_.end());
     77   const std::string& event_name = id_to_event_name_[id];
     78   return event_matchers_[event_name][id]->event_matcher();
     79 }
     80 
     81 const std::string& EventFilter::GetEventName(MatcherID id) {
     82   DCHECK(id_to_event_name_.find(id) != id_to_event_name_.end());
     83   return id_to_event_name_[id];
     84 }
     85 
     86 bool EventFilter::CreateConditionSets(
     87     MatcherID id,
     88     EventMatcher* matcher,
     89     URLMatcherConditionSet::Vector* condition_sets) {
     90   if (matcher->GetURLFilterCount() == 0) {
     91     // If there are no URL filters then we want to match all events, so create a
     92     // URLFilter from an empty dictionary.
     93     base::DictionaryValue empty_dict;
     94     return AddDictionaryAsConditionSet(&empty_dict, condition_sets);
     95   }
     96   for (int i = 0; i < matcher->GetURLFilterCount(); i++) {
     97     base::DictionaryValue* url_filter;
     98     if (!matcher->GetURLFilter(i, &url_filter))
     99       return false;
    100     if (!AddDictionaryAsConditionSet(url_filter, condition_sets))
    101       return false;
    102   }
    103   return true;
    104 }
    105 
    106 bool EventFilter::AddDictionaryAsConditionSet(
    107     base::DictionaryValue* url_filter,
    108     URLMatcherConditionSet::Vector* condition_sets) {
    109   std::string error;
    110   URLMatcherConditionSet::ID condition_set_id = next_condition_set_id_++;
    111   condition_sets->push_back(URLMatcherFactory::CreateFromURLFilterDictionary(
    112       url_matcher_.condition_factory(),
    113       url_filter,
    114       condition_set_id,
    115       &error));
    116   if (!error.empty()) {
    117     LOG(ERROR) << "CreateFromURLFilterDictionary failed: " << error;
    118     url_matcher_.ClearUnusedConditionSets();
    119     condition_sets->clear();
    120     return false;
    121   }
    122   return true;
    123 }
    124 
    125 std::string EventFilter::RemoveEventMatcher(MatcherID id) {
    126   std::map<MatcherID, std::string>::iterator it = id_to_event_name_.find(id);
    127   std::string event_name = it->second;
    128   // EventMatcherEntry's destructor causes the condition set ids to be removed
    129   // from url_matcher_.
    130   event_matchers_[event_name].erase(id);
    131   id_to_event_name_.erase(it);
    132   return event_name;
    133 }
    134 
    135 std::set<EventFilter::MatcherID> EventFilter::MatchEvent(
    136     const std::string& event_name, const EventFilteringInfo& event_info,
    137     int routing_id) {
    138   std::set<MatcherID> matchers;
    139 
    140   EventMatcherMultiMap::iterator it = event_matchers_.find(event_name);
    141   if (it == event_matchers_.end())
    142     return matchers;
    143 
    144   EventMatcherMap& matcher_map = it->second;
    145   GURL url_to_match_against = event_info.has_url() ? event_info.url() : GURL();
    146   std::set<URLMatcherConditionSet::ID> matching_condition_set_ids =
    147       url_matcher_.MatchURL(url_to_match_against);
    148   for (std::set<URLMatcherConditionSet::ID>::iterator it =
    149        matching_condition_set_ids.begin();
    150        it != matching_condition_set_ids.end(); it++) {
    151     std::map<URLMatcherConditionSet::ID, MatcherID>::iterator matcher_id =
    152         condition_set_id_to_event_matcher_id_.find(*it);
    153     if (matcher_id == condition_set_id_to_event_matcher_id_.end()) {
    154       NOTREACHED() << "id not found in condition set map (" << (*it) << ")";
    155       continue;
    156     }
    157     MatcherID id = matcher_id->second;
    158     EventMatcherMap::iterator matcher_entry = matcher_map.find(id);
    159     if (matcher_entry == matcher_map.end()) {
    160       // Matcher must be for a different event.
    161       continue;
    162     }
    163     const EventMatcher* event_matcher = matcher_entry->second->event_matcher();
    164     // The context that installed the event listener should be the same context
    165     // as the one where the event listener is called.
    166     if ((routing_id != MSG_ROUTING_NONE) &&
    167         (event_matcher->GetRoutingID() != routing_id)) {
    168       continue;
    169     }
    170     if (event_matcher->MatchNonURLCriteria(event_info)) {
    171       CHECK(!event_matcher->HasURLFilters() || event_info.has_url());
    172       matchers.insert(id);
    173     }
    174   }
    175 
    176   return matchers;
    177 }
    178 
    179 int EventFilter::GetMatcherCountForEvent(const std::string& name) {
    180   EventMatcherMultiMap::const_iterator it = event_matchers_.find(name);
    181   if (it == event_matchers_.end())
    182     return 0;
    183 
    184   return it->second.size();
    185 }
    186 
    187 }  // namespace extensions
    188