# 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.find_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()