1# Copyright (c) 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. 4 5from autotest_lib.client.common_lib.cros.network import iw_runner 6 7 8# Supported bands 9BAND_2GHZ = '2.4GHz' 10BAND_5GHZ = '5GHz' 11 12# List of valid bands. 13VALID_BANDS = [BAND_2GHZ, BAND_5GHZ] 14 15# List of valid 802.11 protocols (modes). 16MODE_A = 0x01 17MODE_B = 0x02 18MODE_G = 0x04 19MODE_N = 0x08 20MODE_AC = 0x10 21MODE_AUTO = 0x20 22MODE_M = MODE_A | MODE_B | MODE_G # Used for standard maintenance 23MODE_D = MODE_A | MODE_B | MODE_N # International roaming extensions 24MODE_B_G = MODE_B | MODE_G 25MODE_B_G_N = MODE_B | MODE_G | MODE_N 26MODE_AC_N = MODE_AC | MODE_N 27MODE_A_N = MODE_A | MODE_N 28 29# List of valid modes. 30VALID_MODES = [MODE_A, MODE_AC, MODE_AUTO, MODE_B, MODE_D, MODE_G, MODE_M, 31 MODE_N, MODE_B_G, MODE_B_G_N, MODE_A_N, MODE_AC_N] 32VALID_2GHZ_MODES = [MODE_B, MODE_G, MODE_N, MODE_B_G, MODE_B_G_N] 33VALID_5GHZ_MODES = [MODE_A, MODE_AC, MODE_N, MODE_A_N, MODE_AC_N] 34 35# Supported security types 36SECURITY_TYPE_DISABLED = iw_runner.SECURITY_OPEN 37SECURITY_TYPE_WEP = iw_runner.SECURITY_WEP 38SECURITY_TYPE_WPAPSK = iw_runner.SECURITY_WPA 39SECURITY_TYPE_WPA2PSK = iw_runner.SECURITY_WPA2 40# Mixed mode security is wpa/wpa2 41SECURITY_TYPE_MIXED = iw_runner.SECURITY_MIXED 42 43WEP_AUTHENTICATION_OPEN = object() 44WEP_AUTHENTICATION_SHARED = object() 45 46# List of valid securities. 47# TODO (krisr) the configurators do not support WEP at this time. 48VALID_SECURITIES = [SECURITY_TYPE_DISABLED, 49 SECURITY_TYPE_WPAPSK, 50 SECURITY_TYPE_WPA2PSK, 51 SECURITY_TYPE_MIXED, 52 SECURITY_TYPE_WEP] 53 54# List of valid channels. 55VALID_2GHZ_CHANNELS = range(1,15) 56VALID_5GHZ_CHANNELS = [36, 40, 44, 48, 149, 153, 157, 161, 165] 57 58# Frequency to channel conversion table 59CHANNEL_TABLE = {2412: 1, 2417: 2, 2422: 3, 60 2427: 4, 2432: 5, 2437: 6, 61 2442: 7, 2447: 8, 2452: 9, 62 2457: 10, 2462: 11, 2467: 12, 63 2472: 13, 2484: 14, 5180: 36, 64 5200: 40, 5220: 44, 5240: 48, 65 5745: 149, 5765: 153, 5785: 157, 66 5805: 161, 5825: 165} 67 68# This only works because the frequency table is one to one 69# for channels/frequencies. 70FREQUENCY_TABLE = dict((v,k) for k,v in CHANNEL_TABLE.iteritems()) 71 72# Configurator type 73CONFIGURATOR_STATIC = 1 74CONFIGURATOR_DYNAMIC = 2 75CONFIGURATOR_ANY = 3 76 77# Default values 78DEFAULT_BAND = BAND_2GHZ 79 80DEFAULT_2GHZ_MODE = MODE_N 81DEFAULT_5GHZ_MODE = MODE_AC_N 82 83DEFAULT_SECURITY_TYPE = SECURITY_TYPE_WPA2PSK 84 85DEFAULT_2GHZ_CHANNEL = 6 86DEFAULT_5GHZ_CHANNEL = 149 87 88# Convenience method to convert modes and bands to human readable strings. 89def band_string_for_band(band): 90 """Returns a human readable string of the band 91 92 @param band: band object 93 @returns: string representation of the band 94 """ 95 if band == BAND_2GHZ: 96 return '2.4 GHz' 97 elif band == BAND_5GHZ: 98 return '5 GHz' 99 100 101def mode_string_for_mode(mode): 102 """Returns a human readable string of the mode. 103 104 @param mode: integer, the mode to convert. 105 @returns: string representation of the mode 106 """ 107 string_table = {MODE_A:'a', MODE_AC:'ac', MODE_B:'b', MODE_G:'g', 108 MODE_N:'n'} 109 110 if mode == MODE_AUTO: 111 return 'Auto' 112 total = 0 113 string = '' 114 for current_mode in sorted(string_table.keys()): 115 i = current_mode & mode 116 total = total | i 117 if i in string_table: 118 string = string + string_table[i] + '/' 119 if total == MODE_M: 120 string = 'm' 121 elif total == MODE_D: 122 string = 'd' 123 if string[-1] == '/': 124 return string[:-1] 125 return string 126 127 128class APSpec(object): 129 """Object to specify an APs desired capabilities. 130 131 The APSpec object is immutable. All of the parameters are optional. 132 For those not given the defaults listed above will be used. Validation 133 is done on the values to make sure the spec created is valid. If 134 validation fails a ValueError is raised. 135 """ 136 137 138 def __init__(self, visible=True, security=SECURITY_TYPE_WPA2PSK, 139 band=None, mode=None, channel=None, hostnames=None, 140 configurator_type=CONFIGURATOR_ANY, 141 # lab_ap set to true means the AP must be in the lab; 142 # if it set to false the AP is outside of the lab. 143 lab_ap=True): 144 super(APSpec, self).__init__() 145 self._visible = visible 146 self._security = security 147 self._mode = mode 148 self._channel = channel 149 self._hostnames = hostnames 150 self._configurator_type = configurator_type 151 self._lab_ap = lab_ap 152 self._webdriver_hostname = None 153 154 if not self._channel and (self._mode == MODE_N or not self._mode): 155 if band == BAND_2GHZ or not band: 156 self._channel = DEFAULT_2GHZ_CHANNEL 157 if not self._mode: 158 self._mode = DEFAULT_2GHZ_MODE 159 elif band == BAND_5GHZ: 160 self._channel = DEFAULT_5GHZ_CHANNEL 161 if not self._mode: 162 self._mode = DEFAULT_5GHZ_MODE 163 else: 164 raise ValueError('Invalid Band.') 165 166 self._validate_channel_and_mode() 167 168 if ((band == BAND_2GHZ and self._mode not in VALID_2GHZ_MODES) or 169 (band == BAND_5GHZ and self._mode not in VALID_5GHZ_MODES) or 170 (band == BAND_2GHZ and self._channel not in VALID_2GHZ_CHANNELS) or 171 (band == BAND_5GHZ and self._channel not in VALID_5GHZ_CHANNELS)): 172 raise ValueError('Conflicting band and modes/channels.') 173 174 self._validate_security() 175 176 177 def __str__(self): 178 return ('AP Specification:\n' 179 'visible=%r\n' 180 'security=%s\n' 181 'band=%s\n' 182 'mode=%s\n' 183 'channel=%d\n' 184 'password=%s' % (self._visible, self._security, 185 band_string_for_band(self.band), 186 mode_string_for_mode(self._mode), 187 self._channel, self._password)) 188 189 190 @property 191 def password(self): 192 """Returns the password for password supported secured networks.""" 193 return self._password 194 195 196 197 @property 198 def visible(self): 199 """Returns if the SSID is visible or not.""" 200 return self._visible 201 202 203 @property 204 def security(self): 205 """Returns the type of security.""" 206 return self._security 207 208 209 @property 210 def band(self): 211 """Return the band.""" 212 if self._channel in VALID_2GHZ_CHANNELS: 213 return BAND_2GHZ 214 return BAND_5GHZ 215 216 217 @property 218 def mode(self): 219 """Return the mode.""" 220 return self._mode 221 222 223 @property 224 def channel(self): 225 """Return the channel.""" 226 return self._channel 227 228 229 @property 230 def frequency(self): 231 """Return the frequency equivalent of the channel.""" 232 return FREQUENCY_TABLE[self._channel] 233 234 235 @property 236 def hostnames(self): 237 """Return the hostnames; this may be None.""" 238 return self._hostnames 239 240 241 @property 242 def configurator_type(self): 243 """Returns the configurator type.""" 244 return self._configurator_type 245 246 247 @property 248 def lab_ap(self): 249 """Returns if the AP should be in the lab or not.""" 250 return self._lab_ap 251 252 253 @property 254 def webdriver_hostname(self): 255 """Returns locked webdriver hostname.""" 256 return self._webdriver_hostname 257 258 259 @webdriver_hostname.setter 260 def webdriver_hostname(self, value): 261 """Sets webdriver_hostname to locked instance. 262 263 @param value: locked webdriver hostname 264 265 """ 266 self._webdriver_hostname = value 267 268 269 def _validate_channel_and_mode(self): 270 """Validates the channel and mode selected are correct. 271 272 raises ValueError: if the channel or mode selected is invalid 273 """ 274 if self._channel and self._mode: 275 if ((self._channel in VALID_2GHZ_CHANNELS and 276 self._mode not in VALID_2GHZ_MODES) or 277 (self._channel in VALID_5GHZ_CHANNELS and 278 self._mode not in VALID_5GHZ_MODES)): 279 raise ValueError('Conflicting mode/channel has been selected.') 280 elif self._channel: 281 if self._channel in VALID_2GHZ_CHANNELS: 282 self._mode = DEFAULT_2GHZ_MODE 283 elif self._channel in VALID_5GHZ_CHANNELS: 284 self._mode = DEFAULT_5GHZ_MODE 285 else: 286 raise ValueError('Invalid channel passed.') 287 else: 288 if self._mode in VALID_2GHZ_MODES: 289 self._channel = DEFAULT_2GHZ_CHANNEL 290 elif self._mode in VALID_5GHZ_MODES: 291 self._channel = DEFAULT_5GHZ_CHANNEL 292 else: 293 raise ValueError('Invalid mode passed.') 294 295 296 def _validate_security(self): 297 """Sets a password for security settings that need it. 298 299 raises ValueError: if the security setting passed is invalid. 300 """ 301 if self._security == SECURITY_TYPE_DISABLED: 302 self._password = None 303 elif (self._security == SECURITY_TYPE_WPAPSK or 304 self._security == SECURITY_TYPE_WPA2PSK or 305 self._security == SECURITY_TYPE_MIXED): 306 self._password = 'chromeos' 307 elif (self._security==SECURITY_TYPE_WEP): 308 self._password = 'chros' 309 else: 310 raise ValueError('Invalid security passed.') 311