Home | History | Annotate | Download | only in networking
      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