1 # Copyright (c) 2013 The Chromium OS 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 import dbus 6 import logging 7 import subprocess 8 import time 9 10 from autotest_lib.client.cros.networking import shill_proxy 11 12 13 class WifiProxy(shill_proxy.ShillProxy): 14 """Wrapper around shill dbus interface used by wifi tests.""" 15 16 17 def set_logging_for_wifi_test(self): 18 """Set the logging in shill for a test of wifi technology. 19 20 Set the log level to |ShillProxy.LOG_LEVEL_FOR_TEST| and the log scopes 21 to the ones defined in |ShillProxy.LOG_SCOPES_FOR_TEST| for 22 |ShillProxy.TECHNOLOGY_WIFI|. 23 24 """ 25 self.set_logging_for_test(self.TECHNOLOGY_WIFI) 26 27 28 def remove_all_wifi_entries(self): 29 """Iterate over all pushed profiles and remove WiFi entries.""" 30 profiles = self.get_profiles() 31 for profile in profiles: 32 profile_properties = profile.GetProperties(utf8_strings=True) 33 entries = profile_properties[self.PROFILE_PROPERTY_ENTRIES] 34 for entry_id in entries: 35 try: 36 entry = profile.GetEntry(entry_id) 37 except dbus.exceptions.DBusException as e: 38 logging.error('Unable to retrieve entry %s', entry_id) 39 continue 40 if entry[self.ENTRY_FIELD_TYPE] == 'wifi': 41 profile.DeleteEntry(entry_id) 42 43 44 def configure_wifi_service(self, ssid, security, security_parameters={}, 45 save_credentials=True, station_type=None, 46 hidden_network=False, guid=None, 47 autoconnect=None): 48 """Configure a WiFi service. 49 50 @param ssid string name of network to connect to. 51 @param security string type of security used in network (e.g. psk) 52 @param security_parameters dict of service property/value pairs that 53 make up the credentials and settings for the given security 54 type (e.g. the passphrase for psk security). 55 @param save_credentials bool True if we should save EAP credentials. 56 @param station_type string one of SUPPORTED_WIFI_STATION_TYPES. 57 @param hidden_network bool True when the SSID is not broadcasted. 58 @param guid string unique identifier for network. 59 @param autoconnect bool or None. None indicates that this should not 60 be set one way or the other, while a boolean indicates a desired 61 value. 62 63 """ 64 # |mode| is derived from the station type we're attempting to join. It 65 # does not refer to the 802.11x (802.11a/b/g/n) type. It refers to a 66 # shill connection mode. 67 mode = self.SUPPORTED_WIFI_STATION_TYPES[station_type] 68 config_params = {self.SERVICE_PROPERTY_TYPE: 'wifi', 69 self.SERVICE_PROPERTY_HIDDEN: hidden_network, 70 self.SERVICE_PROPERTY_SSID: ssid, 71 self.SERVICE_PROPERTY_SECURITY_CLASS: security, 72 self.SERVICE_PROPERTY_MODE: mode} 73 if autoconnect is not None: 74 config_params[self.SERVICE_PROPERTY_AUTOCONNECT] = autoconnect 75 config_params.update(security_parameters) 76 if guid is not None: 77 config_params[self.SERVICE_PROPERTY_GUID] = guid 78 try: 79 self.configure_service(config_params) 80 except dbus.exceptions.DBusException as e: 81 logging.error('Caught an error while configuring a WiFi ' 82 'service: %r', e) 83 return False 84 85 logging.info('Configured service: %s', ssid) 86 return True 87 88 89 def connect_to_wifi_network(self, 90 ssid, 91 security, 92 security_parameters, 93 save_credentials, 94 station_type=None, 95 hidden_network=False, 96 guid=None, 97 autoconnect=None, 98 discovery_timeout_seconds=15, 99 association_timeout_seconds=15, 100 configuration_timeout_seconds=15): 101 """ 102 Connect to a WiFi network with the given association parameters. 103 104 @param ssid string name of network to connect to. 105 @param security string type of security used in network (e.g. psk) 106 @param security_parameters dict of service property/value pairs that 107 make up the credentials and settings for the given security 108 type (e.g. the passphrase for psk security). 109 @param save_credentials bool True if we should save EAP credentials. 110 @param station_type string one of SUPPORTED_WIFI_STATION_TYPES. 111 @param hidden_network bool True when the SSID is not broadcasted. 112 @param guid string unique identifier for network. 113 @param discovery_timeout_seconds float timeout for service discovery. 114 @param association_timeout_seconds float timeout for service 115 association. 116 @param configuration_timeout_seconds float timeout for DHCP 117 negotiations. 118 @param autoconnect: bool or None. None indicates that this should not 119 be set one way or the other, while a boolean indicates a desired 120 value. 121 @return (successful, discovery_time, association_time, 122 configuration_time, reason) 123 where successful is True iff the operation succeeded, *_time is 124 the time spent waiting for each transition, and reason is a string 125 which may contain a meaningful description of failures. 126 127 """ 128 logging.info('Attempting to connect to %s', ssid) 129 service_proxy = None 130 start_time = time.time() 131 discovery_time = -1.0 132 association_time = -1.0 133 configuration_time = -1.0 134 if station_type not in self.SUPPORTED_WIFI_STATION_TYPES: 135 return (False, discovery_time, association_time, 136 configuration_time, 137 'FAIL(Invalid station type specified.)') 138 139 # |mode| is derived from the station type we're attempting to join. It 140 # does not refer to the 802.11x (802.11a/b/g/n) type. It refers to a 141 # shill connection mode. 142 mode = self.SUPPORTED_WIFI_STATION_TYPES[station_type] 143 144 if hidden_network: 145 logging.info('Configuring %s as a hidden network.', ssid) 146 if not self.configure_wifi_service( 147 ssid, security, save_credentials=save_credentials, 148 station_type=station_type, hidden_network=True, 149 autoconnect=autoconnect): 150 return (False, discovery_time, association_time, 151 configuration_time, 152 'FAIL(Failed to configure hidden SSID)') 153 154 logging.info('Configured hidden service: %s', ssid) 155 156 157 logging.info('Discovering...') 158 discovery_params = {self.SERVICE_PROPERTY_TYPE: 'wifi', 159 self.SERVICE_PROPERTY_NAME: ssid, 160 self.SERVICE_PROPERTY_SECURITY_CLASS: security, 161 self.SERVICE_PROPERTY_MODE: mode} 162 while time.time() - start_time < discovery_timeout_seconds: 163 discovery_time = time.time() - start_time 164 service_object = self.find_matching_service(discovery_params) 165 if service_object: 166 try: 167 service_properties = service_object.GetProperties( 168 utf8_strings=True) 169 except dbus.exceptions.DBusException: 170 # This usually means the service handle has become invalid. 171 # Which is sort of like not getting a handle back from 172 # find_matching_service in the first place. 173 continue 174 strength = self.dbus2primitive( 175 service_properties[self.SERVICE_PROPERTY_STRENGTH]) 176 if strength > 0: 177 logging.info('Discovered service: %s. Strength: %r.', 178 ssid, strength) 179 break 180 181 # This is spammy, but shill handles that for us. 182 self.manager.RequestScan('wifi') 183 time.sleep(self.POLLING_INTERVAL_SECONDS) 184 else: 185 return (False, discovery_time, association_time, 186 configuration_time, 'FAIL(Discovery timed out)') 187 188 # At this point, we know |service| is in the service list. Attempt 189 # to connect it, and watch the states roll by. 190 logging.info('Connecting...') 191 try: 192 for service_property, value in security_parameters.iteritems(): 193 service_object.SetProperty(service_property, value) 194 if guid is not None: 195 service_object.SetProperty(self.SERVICE_PROPERTY_GUID, guid) 196 if autoconnect is not None: 197 service_object.SetProperty(self.SERVICE_PROPERTY_AUTOCONNECT, 198 autoconnect) 199 service_object.Connect() 200 logging.info('Called connect on service') 201 except dbus.exceptions.DBusException, e: 202 logging.error('Caught an error while trying to connect: %s', 203 e.get_dbus_message()) 204 return (False, discovery_time, association_time, 205 configuration_time, 'FAIL(Failed to call connect)') 206 207 logging.info('Associating...') 208 result = self.wait_for_property_in( 209 service_object, 210 self.SERVICE_PROPERTY_STATE, 211 ('configuration', 'ready', 'portal', 'online'), 212 association_timeout_seconds) 213 (successful, _, association_time) = result 214 if not successful: 215 return (False, discovery_time, association_time, 216 configuration_time, 'FAIL(Association timed out)') 217 218 logging.info('Associated with service: %s', ssid) 219 220 logging.info('Configuring...') 221 result = self.wait_for_property_in( 222 service_object, 223 self.SERVICE_PROPERTY_STATE, 224 ('ready', 'portal', 'online'), 225 configuration_timeout_seconds) 226 (successful, _, configuration_time) = result 227 if not successful: 228 return (False, discovery_time, association_time, 229 configuration_time, 'FAIL(Configuration timed out)') 230 231 logging.info('Configured service: %s', ssid) 232 233 # Great success! 234 logging.info('Connected to WiFi service.') 235 return (True, discovery_time, association_time, configuration_time, 236 'SUCCESS(Connection successful)') 237 238 239 def disconnect_from_wifi_network(self, ssid, timeout=None): 240 """Disconnect from the specified WiFi network. 241 242 Method will succeed if it observes the specified network in the idle 243 state after calling Disconnect. 244 245 @param ssid string name of network to disconnect. 246 @param timeout float number of seconds to wait for idle. 247 @return tuple(success, duration, reason) where: 248 success is a bool (True on success). 249 duration is a float number of seconds the operation took. 250 reason is a string containing an informative error on failure. 251 252 """ 253 if timeout is None: 254 timeout = self.SERVICE_DISCONNECT_TIMEOUT 255 service_description = {self.SERVICE_PROPERTY_TYPE: 'wifi', 256 self.SERVICE_PROPERTY_NAME: ssid} 257 service = self.find_matching_service(service_description) 258 if service is None: 259 return (False, 260 0.0, 261 'Failed to disconnect from %s, service not found.' % ssid) 262 263 service.Disconnect() 264 result = self.wait_for_property_in(service, 265 self.SERVICE_PROPERTY_STATE, 266 ('idle',), 267 timeout) 268 (successful, final_state, duration) = result 269 message = 'Success.' 270 if not successful: 271 message = ('Failed to disconnect from %s, ' 272 'timed out in state: %s.' % (ssid, final_state)) 273 return (successful, duration, message) 274 275 276 def configure_bgscan(self, interface, method=None, short_interval=None, 277 long_interval=None, signal=None): 278 """Configures bgscan parameters for wpa_supplicant. 279 280 @param interface string name of interface to configure (e.g. 'mlan0'). 281 @param method string bgscan method (e.g. 'none'). 282 @param short_interval int short scanning interval. 283 @param long_interval int normal scanning interval. 284 @param signal int signal threshold. 285 286 """ 287 device = self.find_object('Device', {'Name': interface}) 288 if device is None: 289 logging.error('No device found with name: %s', interface) 290 return False 291 292 attributes = {'ScanInterval': (dbus.UInt16, long_interval), 293 'BgscanMethod': (dbus.String, method), 294 'BgscanShortInterval': (dbus.UInt16, short_interval), 295 'BgscanSignalThreshold': (dbus.Int32, signal)} 296 for k, (type_cast, value) in attributes.iteritems(): 297 if value is None: 298 continue 299 300 # 'default' is defined in: 301 # client/common_lib/cros/network/xmlrpc_datatypes.py 302 # but we don't have access to that file here. 303 if value == 'default': 304 device.ClearProperty(k) 305 else: 306 device.SetProperty(k, type_cast(value)) 307 return True 308 309 310 def get_active_wifi_SSIDs(self): 311 """@return list of string SSIDs with at least one BSS we've scanned.""" 312 properties = self.manager.GetProperties(utf8_strings=True) 313 services = [self.get_dbus_object(self.DBUS_TYPE_SERVICE, path) 314 for path in properties[self.MANAGER_PROPERTY_SERVICES]] 315 wifi_services = [] 316 for service in services: 317 try: 318 service_properties = self.dbus2primitive(service.GetProperties( 319 utf8_strings=True)) 320 except dbus.exceptions.DBusException as e: 321 pass # Probably the service disappeared before GetProperties(). 322 logging.debug('Considering service with properties: %r', 323 service_properties) 324 service_type = service_properties[self.SERVICE_PROPERTY_TYPE] 325 strength = service_properties[self.SERVICE_PROPERTY_STRENGTH] 326 if service_type == 'wifi' and strength > 0: 327 # Note that this may cause terrible things if the SSID 328 # is not a valid ASCII string. 329 ssid = service_properties[self.SERVICE_PROPERTY_HEX_SSID] 330 logging.info('Found active WiFi service: %s', ssid) 331 wifi_services.append(ssid.decode('hex')) 332 return wifi_services 333 334 335 def wait_for_service_states(self, ssid, states, timeout_seconds): 336 """Wait for a service (ssid) to achieve one of a number of states. 337 338 @param ssid string name of network for whose state we're waiting. 339 @param states tuple states for which to wait. 340 @param timeout_seconds seconds to wait for property to be achieved 341 @return tuple(successful, final_value, duration) 342 where successful is True iff we saw one of |states|, final_value 343 is the final state we saw, and duration is how long we waited to 344 see that value. 345 346 """ 347 discovery_params = {self.SERVICE_PROPERTY_TYPE: 'wifi', 348 self.SERVICE_PROPERTY_NAME: ssid} 349 start_time = time.time() 350 service_object = None 351 while time.time() - start_time < timeout_seconds: 352 service_object = self.find_matching_service(discovery_params) 353 if service_object: 354 break 355 356 time.sleep(self.POLLING_INTERVAL_SECONDS) 357 else: 358 logging.error('Timed out waiting for %s states', ssid) 359 return False, 'unknown', timeout_seconds 360 361 return self.wait_for_property_in( 362 service_object, 363 self.SERVICE_PROPERTY_STATE, 364 states, 365 timeout_seconds - (time.time() - start_time)) 366