|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <set>
 | 
					
						
							|  |  |  | #include "wifiManager.hpp"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * We are using a NetworkManager DBUS API : https://developer.gnome.org/NetworkManager/1.26/spec.html
 | 
					
						
							|  |  |  |  * */
 | 
					
						
							|  |  |  |  
 | 
					
						
							|  |  |  | QString nm_path                = "/org/freedesktop/NetworkManager";
 | 
					
						
							|  |  |  | QString nm_settings_path       = "/org/freedesktop/NetworkManager/Settings";
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString nm_iface               = "org.freedesktop.NetworkManager";
 | 
					
						
							|  |  |  | QString props_iface            = "org.freedesktop.DBus.Properties";
 | 
					
						
							|  |  |  | QString nm_settings_iface      = "org.freedesktop.NetworkManager.Settings";
 | 
					
						
							|  |  |  | QString nm_settings_conn_iface = "org.freedesktop.NetworkManager.Settings.Connection";
 | 
					
						
							|  |  |  | QString device_iface           = "org.freedesktop.NetworkManager.Device";
 | 
					
						
							|  |  |  | QString wireless_device_iface  = "org.freedesktop.NetworkManager.Device.Wireless";
 | 
					
						
							|  |  |  | QString ap_iface               = "org.freedesktop.NetworkManager.AccessPoint";
 | 
					
						
							|  |  |  | QString connection_iface       = "org.freedesktop.NetworkManager.Connection.Active";
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString nm_service             = "org.freedesktop.NetworkManager";
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const int state_connected = 100;
 | 
					
						
							|  |  |  | const int state_need_auth = 60;
 | 
					
						
							|  |  |  | const int reason_wrong_password = 8;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T>
 | 
					
						
							|  |  |  | T get_response(QDBusMessage response){
 | 
					
						
							|  |  |  |   QVariant first =  response.arguments().at(0);
 | 
					
						
							|  |  |  |   QDBusVariant dbvFirst = first.value<QDBusVariant>();
 | 
					
						
							|  |  |  |   QVariant vFirst = dbvFirst.variant();
 | 
					
						
							|  |  |  |   return vFirst.value<T>();
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool compare_by_strength(const Network &a, const Network &b){
 | 
					
						
							|  |  |  |   if (a.connected == ConnectedType::CONNECTED) return true;
 | 
					
						
							|  |  |  |   if (b.connected == ConnectedType::CONNECTED) return false;
 | 
					
						
							|  |  |  |   if (a.connected == ConnectedType::CONNECTING) return true;
 | 
					
						
							|  |  |  |   if (b.connected == ConnectedType::CONNECTING) return false;
 | 
					
						
							|  |  |  |   return a.strength > b.strength;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | WifiManager::WifiManager(){
 | 
					
						
							|  |  |  |   qDBusRegisterMetaType<Connection>();
 | 
					
						
							|  |  |  |   connecting_to_network = "";
 | 
					
						
							|  |  |  |   adapter = get_adapter();
 | 
					
						
							|  |  |  |   has_adapter = adapter != "";
 | 
					
						
							|  |  |  |   if(has_adapter){
 | 
					
						
							|  |  |  |     QDBusInterface nm(nm_service, adapter, device_iface, bus);
 | 
					
						
							|  |  |  |     bus.connect(nm_service, adapter, device_iface, "StateChanged", this, SLOT(change(unsigned int, unsigned int, unsigned int)));
 | 
					
						
							|  |  |  |     
 | 
					
						
							|  |  |  |     QDBusInterface device_props(nm_service, adapter, props_iface, bus);
 | 
					
						
							|  |  |  |     QDBusMessage response = device_props.call("Get", device_iface, "State");
 | 
					
						
							|  |  |  |     raw_adapter_state = get_response<uint>(response);
 | 
					
						
							|  |  |  |     change(raw_adapter_state, 0, 0);
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::refreshNetworks(){
 | 
					
						
							|  |  |  |   if (!has_adapter) return;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bus = QDBusConnection::systemBus();
 | 
					
						
							|  |  |  |   seen_networks.clear();
 | 
					
						
							|  |  |  |   seen_ssids.clear();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (Network &network : get_networks()){
 | 
					
						
							|  |  |  |     if(seen_ssids.count(network.ssid)){
 | 
					
						
							|  |  |  |       continue;
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     seen_ssids.push_back(network.ssid);
 | 
					
						
							|  |  |  |     seen_networks.push_back(network);
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QList<Network> WifiManager::get_networks(){
 | 
					
						
							|  |  |  |   QList<Network> r;
 | 
					
						
							|  |  |  |   QDBusInterface nm(nm_service, adapter, wireless_device_iface, bus);
 | 
					
						
							|  |  |  |   QDBusMessage response = nm.call("GetAllAccessPoints");
 | 
					
						
							|  |  |  |   QVariant first =  response.arguments().at(0);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QString active_ap = get_active_ap();
 | 
					
						
							|  |  |  |   const QDBusArgument &args = first.value<QDBusArgument>();
 | 
					
						
							|  |  |  |   args.beginArray();
 | 
					
						
							|  |  |  |   while (!args.atEnd()) {
 | 
					
						
							|  |  |  |     QDBusObjectPath path;
 | 
					
						
							|  |  |  |     args >> path;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QByteArray ssid = get_property(path.path(), "Ssid");
 | 
					
						
							|  |  |  |     unsigned int strength = get_ap_strength(path.path());
 | 
					
						
							|  |  |  |     SecurityType security = getSecurityType(path.path());
 | 
					
						
							|  |  |  |     ConnectedType ctype;
 | 
					
						
							|  |  |  |     if(path.path() != active_ap){
 | 
					
						
							|  |  |  |       ctype = ConnectedType::DISCONNECTED;
 | 
					
						
							|  |  |  |     }else{
 | 
					
						
							|  |  |  |       if(ssid == connecting_to_network){
 | 
					
						
							|  |  |  |         ctype = ConnectedType::CONNECTING;
 | 
					
						
							|  |  |  |       }else{
 | 
					
						
							|  |  |  |         ctype = ConnectedType::CONNECTED;
 | 
					
						
							|  |  |  |       }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     Network network = {path.path(), ssid, strength, ctype, security};
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ssid.length()){
 | 
					
						
							|  |  |  |       r.push_back(network);
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  |   args.endArray();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   std::sort(r.begin(), r.end(), compare_by_strength);
 | 
					
						
							|  |  |  |   return r;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SecurityType WifiManager::getSecurityType(QString path){
 | 
					
						
							|  |  |  |   int sflag = get_property(path, "Flags").toInt();
 | 
					
						
							|  |  |  |   int wpaflag = get_property(path, "WpaFlags").toInt();
 | 
					
						
							|  |  |  |   int rsnflag = get_property(path, "RsnFlags").toInt();
 | 
					
						
							|  |  |  |   int wpa_props = wpaflag | rsnflag;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if(sflag == 0){
 | 
					
						
							|  |  |  |     return SecurityType::OPEN;
 | 
					
						
							|  |  |  |   } else if((sflag & 0x1) && (wpa_props & (0x333) && !(wpa_props & 0x200))) {
 | 
					
						
							|  |  |  |     return SecurityType::WPA;
 | 
					
						
							|  |  |  |   } else {
 | 
					
						
							|  |  |  |     return SecurityType::UNSUPPORTED;
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::connect(Network n){
 | 
					
						
							|  |  |  |   return connect(n, "", "");
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::connect(Network n, QString password){
 | 
					
						
							|  |  |  |   return connect(n, "", password);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::connect(Network n, QString username, QString password){
 | 
					
						
							|  |  |  |   connecting_to_network = n.ssid;
 | 
					
						
							|  |  |  |   QString active_ap = get_active_ap();
 | 
					
						
							|  |  |  |   if(active_ap!="" && active_ap!="/"){
 | 
					
						
							|  |  |  |     deactivate_connections(get_property(active_ap, "Ssid")); //Disconnect from any connected networks 
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  |   clear_connections(n.ssid); //Clear all connections that may already exist to the network we are connecting
 | 
					
						
							|  |  |  |   connect(n.ssid, username, password, n.security_type);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::connect(QByteArray ssid, QString username, QString password, SecurityType security_type){
 | 
					
						
							|  |  |  |   Connection connection;
 | 
					
						
							|  |  |  |   connection["connection"]["type"] = "802-11-wireless";
 | 
					
						
							|  |  |  |   connection["connection"]["uuid"] = QUuid::createUuid().toString().remove('{').remove('}');
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   connection["connection"]["id"] = "OpenPilot connection "+QString::fromStdString(ssid.toStdString());
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   connection["802-11-wireless"]["ssid"] = ssid;
 | 
					
						
							|  |  |  |   connection["802-11-wireless"]["mode"] = "infrastructure";
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if(security_type == SecurityType::WPA){
 | 
					
						
							|  |  |  |     connection["802-11-wireless-security"]["key-mgmt"] = "wpa-psk";
 | 
					
						
							|  |  |  |     connection["802-11-wireless-security"]["auth-alg"] = "open";
 | 
					
						
							|  |  |  |     connection["802-11-wireless-security"]["psk"] = password;
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   connection["ipv4"]["method"] = "auto";
 | 
					
						
							|  |  |  |   connection["ipv6"]["method"] = "ignore";
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QDBusInterface nm_settings(nm_service, nm_settings_path, nm_settings_iface, bus);
 | 
					
						
							|  |  |  |   QDBusReply<QDBusObjectPath> result = nm_settings.call("AddConnection", QVariant::fromValue(connection));
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::deactivate_connections(QString ssid){
 | 
					
						
							|  |  |  |   for(QDBusObjectPath active_connection_raw : get_active_connections()){
 | 
					
						
							|  |  |  |     QString active_connection = active_connection_raw.path();
 | 
					
						
							|  |  |  |     QDBusInterface nm(nm_service, active_connection, props_iface, bus);
 | 
					
						
							|  |  |  |     QDBusObjectPath pth = get_response<QDBusObjectPath>(nm.call("Get", connection_iface, "SpecificObject"));
 | 
					
						
							|  |  |  |     QString Ssid = get_property(pth.path(), "Ssid");
 | 
					
						
							|  |  |  |     if(Ssid == ssid){
 | 
					
						
							|  |  |  |       QDBusInterface nm2(nm_service, nm_path, nm_iface, bus);
 | 
					
						
							|  |  |  |       nm2.call("DeactivateConnection", QVariant::fromValue(active_connection_raw));
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QVector<QDBusObjectPath> WifiManager::get_active_connections(){
 | 
					
						
							|  |  |  |   QDBusInterface nm(nm_service, nm_path, props_iface, bus);
 | 
					
						
							|  |  |  |   QDBusMessage response = nm.call("Get", nm_iface, "ActiveConnections");
 | 
					
						
							|  |  |  |   QDBusArgument arr = get_response<QDBusArgument>(response);
 | 
					
						
							|  |  |  |   QVector<QDBusObjectPath> conns;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QDBusObjectPath path;
 | 
					
						
							|  |  |  |   arr.beginArray();
 | 
					
						
							|  |  |  |   while (!arr.atEnd()){
 | 
					
						
							|  |  |  |     arr >> path;
 | 
					
						
							|  |  |  |     conns.push_back(path);
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  |   return conns;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::clear_connections(QString ssid){
 | 
					
						
							|  |  |  |   QDBusInterface nm(nm_service, nm_settings_path, nm_settings_iface, bus);
 | 
					
						
							|  |  |  |   QDBusMessage response = nm.call("ListConnections");
 | 
					
						
							|  |  |  |   QVariant first =  response.arguments().at(0);
 | 
					
						
							|  |  |  |   const QDBusArgument &args = first.value<QDBusArgument>();
 | 
					
						
							|  |  |  |   args.beginArray();
 | 
					
						
							|  |  |  |   while (!args.atEnd()) {
 | 
					
						
							|  |  |  |     QDBusObjectPath path;
 | 
					
						
							|  |  |  |     args >> path;
 | 
					
						
							|  |  |  |     QDBusInterface nm2(nm_service, path.path(), nm_settings_conn_iface, bus);
 | 
					
						
							|  |  |  |     QDBusMessage response = nm2.call("GetSettings");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QDBusArgument &dbusArg = response.arguments().at(0).value<QDBusArgument>();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QMap<QString,QMap<QString,QVariant> > map;
 | 
					
						
							|  |  |  |     dbusArg >> map;
 | 
					
						
							|  |  |  |     for(QString outer_key : map.keys()) {
 | 
					
						
							|  |  |  |       QMap<QString,QVariant> innerMap = map.value(outer_key);
 | 
					
						
							|  |  |  |       for(QString inner_key : innerMap.keys()) {
 | 
					
						
							|  |  |  |         if(inner_key == "ssid"){
 | 
					
						
							|  |  |  |           QString value = innerMap.value(inner_key).value<QString>();
 | 
					
						
							|  |  |  |           if(value == ssid){
 | 
					
						
							|  |  |  |             nm2.call("Delete");
 | 
					
						
							|  |  |  |           }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |       }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::request_scan(){
 | 
					
						
							|  |  |  |   if (!has_adapter) return;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QDBusInterface nm(nm_service, adapter, wireless_device_iface, bus);
 | 
					
						
							|  |  |  |   nm.call("RequestScan",  QVariantMap());
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint WifiManager::get_wifi_device_state(){
 | 
					
						
							|  |  |  |   QDBusInterface device_props(nm_service, adapter, props_iface, bus);
 | 
					
						
							|  |  |  |   QDBusMessage response = device_props.call("Get", device_iface, "State");
 | 
					
						
							|  |  |  |   uint resp = get_response<uint>(response);
 | 
					
						
							|  |  |  |   return resp;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString WifiManager::get_active_ap(){
 | 
					
						
							|  |  |  |   QDBusInterface device_props(nm_service, adapter, props_iface, bus);
 | 
					
						
							|  |  |  |   QDBusMessage response = device_props.call("Get", wireless_device_iface, "ActiveAccessPoint");
 | 
					
						
							|  |  |  |   QDBusObjectPath r = get_response<QDBusObjectPath>(response);
 | 
					
						
							|  |  |  |   return r.path();
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QByteArray WifiManager::get_property(QString network_path ,QString property){
 | 
					
						
							|  |  |  |   QDBusInterface device_props(nm_service, network_path, props_iface, bus);
 | 
					
						
							|  |  |  |   QDBusMessage response = device_props.call("Get", ap_iface, property);
 | 
					
						
							|  |  |  |   return get_response<QByteArray>(response);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unsigned int WifiManager::get_ap_strength(QString network_path){
 | 
					
						
							|  |  |  |   QDBusInterface device_props(nm_service, network_path, props_iface, bus);
 | 
					
						
							|  |  |  |   QDBusMessage response = device_props.call("Get", ap_iface, "Strength");
 | 
					
						
							|  |  |  |   return get_response<unsigned int>(response);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QString WifiManager::get_adapter(){
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QDBusInterface nm(nm_service, nm_path, nm_iface, bus);
 | 
					
						
							|  |  |  |   QDBusMessage response = nm.call("GetDevices");
 | 
					
						
							|  |  |  |   QVariant first =  response.arguments().at(0);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QString adapter_path = "";
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const QDBusArgument &args = first.value<QDBusArgument>();
 | 
					
						
							|  |  |  |   args.beginArray();
 | 
					
						
							|  |  |  |   while (!args.atEnd()) {
 | 
					
						
							|  |  |  |     QDBusObjectPath path;
 | 
					
						
							|  |  |  |     args >> path;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Get device type
 | 
					
						
							|  |  |  |     QDBusInterface device_props(nm_service, path.path(), props_iface, bus);
 | 
					
						
							|  |  |  |     QDBusMessage response = device_props.call("Get", device_iface, "DeviceType");
 | 
					
						
							|  |  |  |     uint device_type = get_response<uint>(response);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (device_type == 2) { // Wireless
 | 
					
						
							|  |  |  |       adapter_path = path.path();
 | 
					
						
							|  |  |  |       break;
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  |   args.endArray();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return adapter_path;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WifiManager::change(unsigned int new_state,unsigned int previous_state,unsigned int change_reason){
 | 
					
						
							|  |  |  |   raw_adapter_state = new_state;
 | 
					
						
							|  |  |  |   if(new_state == state_need_auth && change_reason == reason_wrong_password){
 | 
					
						
							|  |  |  |     emit wrongPassword(connecting_to_network);
 | 
					
						
							|  |  |  |   }else if(new_state == state_connected){
 | 
					
						
							|  |  |  |     connecting_to_network="";
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | }
 |