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.
5 #include "chrome/browser/extensions/api/mdns/mdns_api.h"
9 #include "base/lazy_instance.h"
10 #include "chrome/browser/extensions/extension_service.h"
11 #include "chrome/common/extensions/api/mdns.h"
12 #include "extensions/browser/extension_system.h"
14 namespace extensions {
16 namespace mdns = api::mdns;
20 // Whitelisted mDNS service types.
21 const char kCastServiceType[] = "_googlecast._tcp.local";
22 const char kPrivetServiceType[] = "_privet._tcp.local";
23 const char kTestServiceType[] = "_testing._tcp.local";
25 bool IsServiceTypeWhitelisted(const std::string& service_type) {
26 return service_type == kCastServiceType ||
27 service_type == kPrivetServiceType ||
28 service_type == kTestServiceType;
33 MDnsAPI::MDnsAPI(Profile* profile) : profile_(profile) {
35 ExtensionSystem::Get(profile)->event_router()->RegisterObserver(
36 this, mdns::OnServiceList::kEventName);
40 if (dns_sd_registry_.get()) {
41 dns_sd_registry_->RemoveObserver(this);
46 MDnsAPI* MDnsAPI::Get(Profile* profile) {
47 return ProfileKeyedAPIFactory<MDnsAPI>::GetForProfile(profile);
50 static base::LazyInstance<ProfileKeyedAPIFactory<MDnsAPI> > g_factory =
51 LAZY_INSTANCE_INITIALIZER;
54 ProfileKeyedAPIFactory<MDnsAPI>* MDnsAPI::GetFactoryInstance() {
55 return g_factory.Pointer();
58 void MDnsAPI::SetDnsSdRegistryForTesting(
59 scoped_ptr<DnsSdRegistry> dns_sd_registry) {
60 dns_sd_registry_ = dns_sd_registry.Pass();
63 DnsSdRegistry* MDnsAPI::dns_sd_registry() {
64 DCHECK(thread_checker_.CalledOnValidThread());
65 if (!dns_sd_registry_.get()) {
66 dns_sd_registry_.reset(new extensions::DnsSdRegistry());
67 dns_sd_registry_->AddObserver(this);
69 return dns_sd_registry_.get();
72 void MDnsAPI::OnListenerAdded(const EventListenerInfo& details) {
73 DCHECK(thread_checker_.CalledOnValidThread());
74 UpdateMDnsListeners(details);
77 void MDnsAPI::OnListenerRemoved(const EventListenerInfo& details) {
78 DCHECK(thread_checker_.CalledOnValidThread());
79 UpdateMDnsListeners(details);
82 void MDnsAPI::UpdateMDnsListeners(const EventListenerInfo& details) {
83 std::set<std::string> new_service_types;
85 // Check all listeners for service type filers.
86 const EventListenerMap::ListenerList& listeners =
87 extensions::ExtensionSystem::Get(profile_)->event_router()->
88 listeners().GetEventListenersByName(details.event_name);
89 for (EventListenerMap::ListenerList::const_iterator it = listeners.begin();
90 it != listeners.end(); ++it) {
91 base::DictionaryValue* filter = ((*it)->filter.get());
93 std::string filter_value;
94 filter->GetStringASCII(kEventFilterServiceTypeKey, &filter_value);
95 if (filter_value.empty())
97 new_service_types.insert(filter_value);
100 // Find all the added and removed service types since last update.
101 std::set<std::string> added_service_types =
102 base::STLSetDifference<std::set<std::string> >(
103 new_service_types, service_types_);
104 std::set<std::string> removed_service_types =
105 base::STLSetDifference<std::set<std::string> >(
106 service_types_, new_service_types);
108 // Update the registry.
109 DnsSdRegistry* registry = dns_sd_registry();
110 for (std::set<std::string>::iterator it = added_service_types.begin();
111 it != added_service_types.end(); ++it) {
112 if (IsServiceTypeWhitelisted(*it))
113 registry->RegisterDnsSdListener(*it);
115 for (std::set<std::string>::iterator it = removed_service_types.begin();
116 it != removed_service_types.end(); ++it) {
117 if (IsServiceTypeWhitelisted(*it))
118 registry->UnregisterDnsSdListener(*it);
121 service_types_ = new_service_types;
124 void MDnsAPI::OnDnsSdEvent(const std::string& service_type,
125 const DnsSdRegistry::DnsSdServiceList& services) {
126 DCHECK(thread_checker_.CalledOnValidThread());
128 std::vector<linked_ptr<mdns::MDnsService> > args;
129 for (DnsSdRegistry::DnsSdServiceList::const_iterator it = services.begin();
130 it != services.end(); ++it) {
131 linked_ptr<mdns::MDnsService> mdns_service =
132 make_linked_ptr(new mdns::MDnsService);
133 mdns_service->service_name = (*it).service_name;
134 mdns_service->service_host_port = (*it).service_host_port;
135 mdns_service->ip_address = (*it).ip_address;
136 mdns_service->service_data = (*it).service_data;
137 args.push_back(mdns_service);
140 scoped_ptr<base::ListValue> results = mdns::OnServiceList::Create(args);
141 scoped_ptr<Event> event(
142 new Event(mdns::OnServiceList::kEventName, results.Pass()));
143 event->restrict_to_browser_context = profile_;
144 event->filter_info.SetServiceType(service_type);
146 VLOG(1) << "Broadcasting OnServiceList event: " << event.get();
148 // TODO(justinlin): To avoid having listeners without filters getting all
149 // events, modify API to have this event require filters.
150 extensions::ExtensionSystem::Get(profile_)->event_router()->
151 BroadcastEvent(event.Pass());
154 } // namespace extensions