134 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			134 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Python
		
	
	
	
| # Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
 | |
| # Use of this source code is governed by a BSD-style license that can be
 | |
| # found in the LICENSE file.
 | |
| 
 | |
| import dbus
 | |
| import logging
 | |
| import random
 | |
| import time
 | |
| 
 | |
| from autotest_lib.client.bin import test
 | |
| from autotest_lib.client.common_lib import error
 | |
| from autotest_lib.client.cros.networking import cellular_proxy
 | |
| from autotest_lib.client.cros.networking import shill_context
 | |
| from autotest_lib.client.cros.networking import shill_proxy
 | |
| 
 | |
| 
 | |
| class cellular_SafetyDance(test.test):
 | |
|     """
 | |
|     Stress tests all connection manager 3G operations.
 | |
| 
 | |
|     This test runs a long series of 3G operations in pseudorandom order. All of
 | |
|     these 3G operations must return a convincing result (EINPROGRESS or no
 | |
|     error).
 | |
| 
 | |
|     """
 | |
|     version = 1
 | |
| 
 | |
|     def _filterexns(self, fn):
 | |
|         v = None
 | |
|         try:
 | |
|             v = fn()
 | |
|         except dbus.exceptions.DBusException, error:
 | |
|             if error.get_dbus_name() in self.okerrors:
 | |
|                 return v, error.get_dbus_message()
 | |
|             else:
 | |
|                 raise error
 | |
|         return v, ''
 | |
| 
 | |
|     def _enable(self):
 | |
|         logging.info('Enable')
 | |
|         self._filterexns(lambda:
 | |
|             self.test_env.shill.manager.EnableTechnology('cellular'))
 | |
| 
 | |
|     def _disable(self):
 | |
|         logging.info('Disable')
 | |
|         self._filterexns(lambda:
 | |
|             self.test_env.shill.manager.DisableTechnology('cellular'))
 | |
| 
 | |
|     def _ignoring(self, reason):
 | |
|         if ('AlreadyConnected' in reason or
 | |
|             'Not connected' in reason or
 | |
|             'Bearer already being connected' in reason or
 | |
|             'Bearer already being disconnected' in reason or
 | |
|             'InProgress' in reason):
 | |
|             return True
 | |
|         if 'NotSupported' in reason:
 | |
|             # We should only ignore this error if we've previously disabled
 | |
|             # cellular technology and the service subsequently disappeared
 | |
|             # when we tried to connect again.
 | |
|             return not self.test_env.shill.find_cellular_service_object()
 | |
|         return False
 | |
| 
 | |
|     def _connect(self):
 | |
|         logging.info('Connect')
 | |
|         try:
 | |
|             service = self.test_env.shill.wait_for_cellular_service_object(
 | |
|                     timeout_seconds=5)
 | |
|         except shill_proxy.ShillProxyError:
 | |
|             return
 | |
| 
 | |
|         success, reason = self._filterexns(lambda:
 | |
|                 self.test_env.shill.connect_service_synchronous(
 | |
|                         service=service,
 | |
|                         timeout_seconds=
 | |
|                         cellular_proxy.CellularProxy.SERVICE_CONNECT_TIMEOUT))
 | |
|         if not success and not self._ignoring(reason):
 | |
|             raise error.TestFail('Could not connect: %s' % reason)
 | |
| 
 | |
|     def _disconnect(self):
 | |
|         logging.info('Disconnect')
 | |
|         try:
 | |
|             service = self.test_env.shill.wait_for_cellular_service_object(
 | |
|                     timeout_seconds=5)
 | |
|         except shill_proxy.ShillProxyError:
 | |
|             return
 | |
| 
 | |
|         success, reason = self._filterexns(lambda:
 | |
|                 self.test_env.shill.disconnect_service_synchronous(
 | |
|                         service=service,
 | |
|                         timeout_seconds=
 | |
|                         cellular_proxy.CellularProxy.
 | |
|                         SERVICE_DISCONNECT_TIMEOUT))
 | |
|         if not success and not self._ignoring(reason):
 | |
|             raise error.TestFail('Could not disconnect: %s' % reason)
 | |
| 
 | |
|     def _op(self):
 | |
|         n = random.randint(0, len(self.ops) - 1)
 | |
|         self.ops[n]()
 | |
|         time.sleep(random.randint(5, 20) / 10.0)
 | |
| 
 | |
|     def _run_once_internal(self, ops=30, seed=None):
 | |
|         if not seed:
 | |
|             seed = int(time.time())
 | |
|         self.okerrors = [
 | |
|             'org.chromium.flimflam.Error.InProgress',
 | |
|             'org.chromium.flimflam.Error.AlreadyConnected',
 | |
|             'org.chromium.flimflam.Error.AlreadyEnabled',
 | |
|             'org.chromium.flimflam.Error.AlreadyDisabled'
 | |
|         ]
 | |
|         self.ops = [ self._enable,
 | |
|                      self._disable,
 | |
|                      self._connect,
 | |
|                      self._disconnect ]
 | |
|         self.device = self.test_env.shill.find_cellular_device_object()
 | |
|         if not self.device:
 | |
|             raise error.TestFail('Could not find cellular device.')
 | |
| 
 | |
|         # Start in a disabled state.
 | |
|         self._disable()
 | |
|         logging.info('Seed: %d', seed)
 | |
|         random.seed(seed)
 | |
|         for _ in xrange(ops):
 | |
|             self._op()
 | |
| 
 | |
|     def run_once(self, test_env, ops=30, seed=None):
 | |
|         self.test_env = test_env
 | |
|         with test_env, shill_context.ServiceAutoConnectContext(
 | |
|                 test_env.shill.wait_for_cellular_service_object, False):
 | |
|             self._run_once_internal(ops, seed)
 | |
| 
 | |
|             # Enable device to restore autoconnect settings.
 | |
|             self._enable()
 | |
|             test_env.shill.wait_for_cellular_service_object()
 |