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"""A collection of context managers for working with shill objects.""" 6 7import errno 8import logging 9import os 10 11from contextlib import contextmanager 12 13from autotest_lib.client.common_lib import error 14from autotest_lib.client.common_lib import utils 15from autotest_lib.client.cros.networking import shill_proxy 16 17SHILL_START_LOCK_PATH = '/var/lock/shill-start.lock' 18 19class ContextError(Exception): 20 """An error raised by a context managers dealing with shill objects.""" 21 pass 22 23 24class ServiceAutoConnectContext(object): 25 """A context manager for overriding a service's 'AutoConnect' property. 26 27 As the service object of the same service may change during the lifetime 28 of the context, this context manager does not take a service object at 29 construction. Instead, it takes a |get_service| function at construction, 30 which it invokes to obtain a service object when entering and exiting the 31 context. It is assumed that |get_service| always returns a service object 32 that refers to the same service. 33 34 Usage: 35 def get_service(): 36 # Some function that returns a service object. 37 38 with ServiceAutoConnectContext(get_service, False): 39 # Within this context, the 'AutoConnect' property of the service 40 # returned by |get_service| is temporarily set to False if it's 41 # initially set to True. The property is restored to its initial 42 # value after the context ends. 43 44 """ 45 def __init__(self, get_service, autoconnect): 46 self._get_service = get_service 47 self._autoconnect = autoconnect 48 self._initial_autoconnect = None 49 50 51 def __enter__(self): 52 service = self._get_service() 53 if service is None: 54 raise ContextError('Could not obtain a service object.') 55 56 # Always set the AutoConnect property even if the requested value 57 # is the same so that shill will retain the AutoConnect property, else 58 # shill may override it. 59 service_properties = service.GetProperties() 60 self._initial_autoconnect = shill_proxy.ShillProxy.dbus2primitive( 61 service_properties[ 62 shill_proxy.ShillProxy.SERVICE_PROPERTY_AUTOCONNECT]) 63 logging.info('ServiceAutoConnectContext: change autoconnect to %s', 64 self._autoconnect) 65 service.SetProperty( 66 shill_proxy.ShillProxy.SERVICE_PROPERTY_AUTOCONNECT, 67 self._autoconnect) 68 69 # Make sure the cellular service gets persisted by taking it out of 70 # the ephemeral profile. 71 if not service_properties[ 72 shill_proxy.ShillProxy.SERVICE_PROPERTY_PROFILE]: 73 shill = shill_proxy.ShillProxy.get_proxy() 74 manager_properties = shill.manager.GetProperties(utf8_strings=True) 75 active_profile = manager_properties[ 76 shill_proxy.ShillProxy.MANAGER_PROPERTY_ACTIVE_PROFILE] 77 logging.info('ServiceAutoConnectContext: change cellular service ' 78 'profile to %s', active_profile) 79 service.SetProperty( 80 shill_proxy.ShillProxy.SERVICE_PROPERTY_PROFILE, 81 active_profile) 82 83 return self 84 85 86 def __exit__(self, exc_type, exc_value, traceback): 87 if self._initial_autoconnect != self._autoconnect: 88 service = self._get_service() 89 if service is None: 90 raise ContextError('Could not obtain a service object.') 91 92 logging.info('ServiceAutoConnectContext: restore autoconnect to %s', 93 self._initial_autoconnect) 94 service.SetProperty( 95 shill_proxy.ShillProxy.SERVICE_PROPERTY_AUTOCONNECT, 96 self._initial_autoconnect) 97 return False 98 99 100 @property 101 def autoconnect(self): 102 """AutoConnect property value within this context.""" 103 return self._autoconnect 104 105 106 @property 107 def initial_autoconnect(self): 108 """Initial AutoConnect property value when entering this context.""" 109 return self._initial_autoconnect 110 111 112@contextmanager 113def stopped_shill(): 114 """A context for executing code which requires shill to be stopped. 115 116 This context stops shill on entry to the context, and starts shill 117 before exit from the context. This context further guarantees that 118 shill will be not restarted by recover_duts, while this context is 119 active. 120 121 Note that the no-restart guarantee applies only if the user of 122 this context completes with a 'reasonable' amount of time. In 123 particular: if networking is unavailable for 15 minutes or more, 124 recover_duts will reboot the DUT. 125 126 """ 127 def get_lock_holder(lock_path): 128 lock_holder = os.readlink(lock_path) 129 try: 130 os.stat(lock_holder) 131 return lock_holder # stat() success -> valid link -> locker alive 132 except OSError as e: 133 if e.errno == errno.ENOENT: # dangling link -> locker is gone 134 return None 135 else: 136 raise 137 138 our_proc_dir = '/proc/%d/' % os.getpid() 139 try: 140 os.symlink(our_proc_dir, SHILL_START_LOCK_PATH) 141 except OSError as e: 142 if e.errno != errno.EEXIST: 143 raise 144 lock_holder = get_lock_holder(SHILL_START_LOCK_PATH) 145 if lock_holder is not None: 146 raise error.TestError('Shill start lock held by %s' % lock_holder) 147 os.remove(SHILL_START_LOCK_PATH) 148 os.symlink(our_proc_dir, SHILL_START_LOCK_PATH) 149 150 utils.run('stop shill') 151 yield 152 utils.run('start shill') 153 os.remove(SHILL_START_LOCK_PATH) 154