| // Copyright (c) 2013 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 "chromeos/network/geolocation_handler.h" |
| |
| #include "base/bind.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "base/values.h" |
| #include "chromeos/dbus/dbus_thread_manager.h" |
| #include "chromeos/dbus/shill_manager_client.h" |
| #include "third_party/cros_system_api/dbus/service_constants.h" |
| |
| namespace chromeos { |
| |
| GeolocationHandler::GeolocationHandler() |
| : wifi_enabled_(false), |
| weak_ptr_factory_(this) { |
| } |
| |
| GeolocationHandler::~GeolocationHandler() { |
| ShillManagerClient* manager_client = |
| DBusThreadManager::Get()->GetShillManagerClient(); |
| if (manager_client) |
| manager_client->RemovePropertyChangedObserver(this); |
| } |
| |
| void GeolocationHandler::Init() { |
| ShillManagerClient* manager_client = |
| DBusThreadManager::Get()->GetShillManagerClient(); |
| manager_client->GetProperties( |
| base::Bind(&GeolocationHandler::ManagerPropertiesCallback, |
| weak_ptr_factory_.GetWeakPtr())); |
| manager_client->AddPropertyChangedObserver(this); |
| } |
| |
| bool GeolocationHandler::GetWifiAccessPoints( |
| WifiAccessPointVector* access_points, int64* age_ms) { |
| if (!wifi_enabled_) |
| return false; |
| // Always request updated access points. |
| RequestWifiAccessPoints(); |
| // If no data has been received, return false. |
| if (geolocation_received_time_.is_null()) |
| return false; |
| if (access_points) |
| *access_points = wifi_access_points_; |
| if (age_ms) { |
| base::TimeDelta dtime = base::Time::Now() - geolocation_received_time_; |
| *age_ms = dtime.InMilliseconds(); |
| } |
| return true; |
| } |
| |
| void GeolocationHandler::OnPropertyChanged(const std::string& key, |
| const base::Value& value) { |
| HandlePropertyChanged(key, value); |
| } |
| |
| //------------------------------------------------------------------------------ |
| // Private methods |
| |
| void GeolocationHandler::ManagerPropertiesCallback( |
| DBusMethodCallStatus call_status, |
| const base::DictionaryValue& properties) { |
| const base::Value* value = NULL; |
| if (properties.Get(shill::kEnabledTechnologiesProperty, &value) && value) |
| HandlePropertyChanged(shill::kEnabledTechnologiesProperty, *value); |
| } |
| |
| void GeolocationHandler::HandlePropertyChanged(const std::string& key, |
| const base::Value& value) { |
| if (key != shill::kEnabledTechnologiesProperty) |
| return; |
| const base::ListValue* technologies = NULL; |
| if (!value.GetAsList(&technologies) || !technologies) |
| return; |
| bool wifi_was_enabled = wifi_enabled_; |
| wifi_enabled_ = false; |
| for (base::ListValue::const_iterator iter = technologies->begin(); |
| iter != technologies->end(); ++iter) { |
| std::string technology; |
| (*iter)->GetAsString(&technology); |
| if (technology == shill::kTypeWifi) { |
| wifi_enabled_ = true; |
| break; |
| } |
| } |
| if (!wifi_was_enabled && wifi_enabled_) |
| RequestWifiAccessPoints(); // Request initial location data. |
| } |
| |
| void GeolocationHandler::RequestWifiAccessPoints() { |
| DBusThreadManager::Get()->GetShillManagerClient()->GetNetworksForGeolocation( |
| base::Bind(&GeolocationHandler::GeolocationCallback, |
| weak_ptr_factory_.GetWeakPtr())); |
| } |
| |
| void GeolocationHandler::GeolocationCallback( |
| DBusMethodCallStatus call_status, |
| const base::DictionaryValue& properties) { |
| if (call_status != DBUS_METHOD_CALL_SUCCESS) { |
| LOG(ERROR) << "Failed to get Geolocation data: " << call_status; |
| return; |
| } |
| wifi_access_points_.clear(); |
| if (properties.empty()) |
| return; // No enabled devices, don't update received time. |
| |
| // Dictionary<device_type, entry_list> |
| for (base::DictionaryValue::Iterator iter(properties); |
| !iter.IsAtEnd(); iter.Advance()) { |
| const base::ListValue* entry_list = NULL; |
| if (!iter.value().GetAsList(&entry_list)) { |
| LOG(WARNING) << "Geolocation dictionary value not a List: " << iter.key(); |
| continue; |
| } |
| // List[Dictionary<key, value_str>] |
| for (size_t i = 0; i < entry_list->GetSize(); ++i) { |
| const base::DictionaryValue* entry = NULL; |
| if (!entry_list->GetDictionary(i, &entry) || !entry) { |
| LOG(WARNING) << "Geolocation list value not a Dictionary: " << i; |
| continue; |
| } |
| // Docs: developers.google.com/maps/documentation/business/geolocation |
| WifiAccessPoint wap; |
| entry->GetString(shill::kGeoMacAddressProperty, &wap.mac_address); |
| std::string age_str; |
| if (entry->GetString(shill::kGeoAgeProperty, &age_str)) { |
| int64 age_ms; |
| if (base::StringToInt64(age_str, &age_ms)) { |
| wap.timestamp = |
| base::Time::Now() - base::TimeDelta::FromMilliseconds(age_ms); |
| } |
| } |
| std::string strength_str; |
| if (entry->GetString(shill::kGeoSignalStrengthProperty, &strength_str)) |
| base::StringToInt(strength_str, &wap.signal_strength); |
| std::string signal_str; |
| if (entry->GetString(shill::kGeoSignalToNoiseRatioProperty, &signal_str)) |
| base::StringToInt(signal_str, &wap.signal_to_noise); |
| std::string channel_str; |
| if (entry->GetString(shill::kGeoChannelProperty, &channel_str)) |
| base::StringToInt(channel_str, &wap.channel); |
| wifi_access_points_.push_back(wap); |
| } |
| } |
| geolocation_received_time_ = base::Time::Now(); |
| } |
| |
| } // namespace chromeos |