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
5import logging
6import random
7import requests
8
9from time import sleep
10
11import common
12from autotest_lib.client.common_lib import utils
13from autotest_lib.server.cros.ap_configurators import \
14    ap_configurator_factory
15from autotest_lib.client.common_lib.cros.network import ap_constants
16from autotest_lib.server.cros.ap_configurators import ap_cartridge
17
18
19# Max number of retry attempts to lock an ap.
20MAX_RETRIES = 3
21CHAOS_URL = 'https://chaos-188802.appspot.com'
22
23
24class ApLocker(object):
25    """Object to keep track of AP lock state.
26
27    @attribute configurator: an APConfigurator object.
28    @attribute to_be_locked: a boolean, True iff ap has not been locked.
29    @attribute retries: an integer, max number of retry attempts to lock ap.
30    """
31
32
33    def __init__(self, configurator, retries):
34        """Initialize.
35
36        @param configurator: an APConfigurator object.
37        @param retries: an integer, max number of retry attempts to lock ap.
38        """
39        self.configurator = configurator
40        self.to_be_locked = True
41        self.retries = retries
42
43
44    def __repr__(self):
45        """@return class name, ap host name, lock status and retries."""
46        return 'class: %s, host name: %s, to_be_locked = %s, retries = %d' % (
47                self.__class__.__name__,
48                self.configurator.host_name,
49                self.to_be_locked,
50                self.retries)
51
52
53def construct_ap_lockers(ap_spec, retries, hostname_matching_only=False,
54                         ap_test_type=ap_constants.AP_TEST_TYPE_CHAOS):
55    """Convert APConfigurator objects to ApLocker objects for locking.
56
57    @param ap_spec: an APSpec object
58    @param retries: an integer, max number of retry attempts to lock ap.
59    @param hostname_matching_only: a boolean, if True matching against
60                                   all other APSpec parameters is not
61                                   performed.
62    @param ap_test_type: Used to determine which type of test we're
63                         currently running (Chaos vs Clique).
64
65    @return a list of ApLocker objects.
66    """
67    ap_lockers_list = []
68    factory = ap_configurator_factory.APConfiguratorFactory(ap_test_type,
69                                                            ap_spec)
70    if hostname_matching_only:
71        for ap in factory.get_aps_by_hostnames(ap_spec.hostnames):
72            ap_lockers_list.append(ApLocker(ap, retries))
73    else:
74        for ap in factory.get_ap_configurators_by_spec(ap_spec):
75            ap_lockers_list.append(ApLocker(ap, retries))
76
77    if not len(ap_lockers_list):
78        logging.error('Found no matching APs to test against for %s', ap_spec)
79
80    logging.debug('Found %d APs', len(ap_lockers_list))
81    return ap_lockers_list
82
83
84class ApBatchLocker(object):
85    """Object to lock/unlock an APConfigurator.
86
87    @attribute SECONDS_TO_SLEEP: an integer, number of seconds to sleep between
88                                 retries.
89    @attribute ap_spec: an APSpec object
90    @attribute retries: an integer, max number of retry attempts to lock ap.
91                        Defaults to MAX_RETRIES.
92    @attribute aps_to_lock: a list of ApLocker objects.
93    @attribute manager: a HostLockManager object, used to lock/unlock APs.
94    """
95
96
97    MIN_SECONDS_TO_SLEEP = 30
98    MAX_SECONDS_TO_SLEEP = 120
99
100
101    def __init__(self, lock_manager, ap_spec, retries=MAX_RETRIES,
102                 hostname_matching_only=False,
103                 ap_test_type=ap_constants.AP_TEST_TYPE_CHAOS):
104        """Initialize.
105
106        @param ap_spec: an APSpec object
107        @param retries: an integer, max number of retry attempts to lock ap.
108                        Defaults to MAX_RETRIES.
109        @param hostname_matching_only : a boolean, if True matching against
110                                        all other APSpec parameters is not
111                                        performed.
112        @param ap_test_type: Used to determine which type of test we're
113                             currently running (Chaos vs Clique).
114        """
115        self.aps_to_lock = construct_ap_lockers(ap_spec, retries,
116                           hostname_matching_only=hostname_matching_only,
117                           ap_test_type=ap_test_type)
118        self.manager = lock_manager
119        self._locked_aps = []
120
121
122    def has_more_aps(self):
123        """@return True iff there is at least one AP to be locked."""
124        return len(self.aps_to_lock) > 0
125
126
127    def lock_ap_in_afe(self, ap_locker):
128        """Locks an AP host in AFE.
129
130        @param ap_locker: an ApLocker object, AP to be locked.
131        @return a boolean, True iff ap_locker is locked.
132        """
133        if not utils.host_is_in_lab_zone(ap_locker.configurator.host_name):
134            ap_locker.to_be_locked = False
135            return True
136
137        if self.manager.lock([ap_locker.configurator.host_name]):
138            self._locked_aps.append(ap_locker)
139            logging.info('locked %s', ap_locker.configurator.host_name)
140            ap_locker.to_be_locked = False
141            return True
142        else:
143            ap_locker.retries -= 1
144            logging.info('%d retries left for %s',
145                         ap_locker.retries,
146                         ap_locker.configurator.host_name)
147            if ap_locker.retries == 0:
148                logging.info('No more retries left. Remove %s from list',
149                             ap_locker.configurator.host_name)
150                ap_locker.to_be_locked = False
151
152        return False
153
154    def lock_ap_in_datastore(self, ap_locker):
155        """Locks an AP host in datastore.
156
157        @param ap_locker: an ApLocker object, AP to be locked.
158        @return a boolean, True iff ap_locker is locked.
159        """
160        if not utils.host_is_in_lab_zone(ap_locker.configurator.host_name):
161            ap_locker.to_be_locked = False
162            return True
163
164        # Begin locking device in datastore.
165        locked_device = requests.put(CHAOS_URL + '/devices/lock', \
166                        json={"hostname":[ap_locker.configurator.host_name], \
167                        "locked_by":"TestRun"})
168        if locked_device.json()['result']:
169            self._locked_aps.append(ap_locker)
170            logging.info('locked %s', ap_locker.configurator.host_name)
171            ap_locker.to_be_locked = False
172            return True
173        else:
174            ap_locker.retries -= 1
175            logging.info('%d retries left for %s',
176                         ap_locker.retries,
177                         ap_locker.configurator.host_name)
178            if ap_locker.retries == 0:
179                logging.info('No more retries left. Remove %s from list',
180                             ap_locker.configurator.host_name)
181                ap_locker.to_be_locked = False
182
183        return False
184
185
186    def get_ap_batch(self, batch_size=ap_cartridge.THREAD_MAX):
187        """Allocates a batch of locked APs.
188
189        @param batch_size: an integer, max. number of aps to lock in one batch.
190                           Defaults to THREAD_MAX in ap_cartridge.py
191        @return a list of APConfigurator objects, locked on AFE.
192        """
193        # We need this while loop to continuously loop over the for loop.
194        # To exit the while loop, we either:
195        #  - locked batch_size number of aps and return them
196        #  - exhausted all retries on all aps in aps_to_lock
197        while len(self.aps_to_lock):
198            ap_batch = []
199
200            for ap_locker in self.aps_to_lock:
201                logging.info('checking %s', ap_locker.configurator.host_name)
202                # TODO(@rjahagir): Change method to datastore.
203                # if self.lock_ap_in_datastore(ap_locker):
204                if self.lock_ap_in_afe(ap_locker):
205                    ap_batch.append(ap_locker.configurator)
206                    if len(ap_batch) == batch_size:
207                        break
208
209            # Remove locked APs from list of APs to process.
210            aps_to_rm = [ap for ap in self.aps_to_lock if not ap.to_be_locked]
211            self.aps_to_lock = list(set(self.aps_to_lock) - set(aps_to_rm))
212            for ap in aps_to_rm:
213                logging.info('Removed %s from self.aps_to_lock',
214                             ap.configurator.host_name)
215            logging.info('Remaining aps to lock = %s',
216                         [ap.configurator.host_name for ap in self.aps_to_lock])
217
218            # Return available APs and retry remaining ones later.
219            if ap_batch:
220                return ap_batch
221
222            # Sleep before next retry.
223            if self.aps_to_lock:
224                seconds_to_sleep = random.randint(self.MIN_SECONDS_TO_SLEEP,
225                                                  self.MAX_SECONDS_TO_SLEEP)
226                logging.info('Sleep %d sec before retry', seconds_to_sleep)
227                sleep(seconds_to_sleep)
228
229        return []
230
231
232    def unlock_one_ap(self, host_name):
233        """Unlock one AP after we're done.
234
235        @param host_name: a string, host name.
236        """
237        for ap_locker in self._locked_aps:
238            if host_name == ap_locker.configurator.host_name:
239                self.manager.unlock(hosts=[host_name])
240                self._locked_aps.remove(ap_locker)
241                return
242
243        logging.error('Tried to unlock a host we have not locked (%s)?',
244                      host_name)
245
246    def unlock_one_ap_in_datastore(self, host_name):
247        """Unlock one AP from datastore after we're done.
248
249        @param host_name: a string, host name.
250        """
251        for ap_locker in self._locked_aps:
252            if host_name == ap_locker.configurator.host_name:
253                # Unlock in datastore
254                unlocked_device = requests.put(CHAOS_URL + '/devices/unlock', \
255                                  json={"hostname":host_name})
256                # TODO: Raise error if unable to unlock.
257                if not unlocked_device.json()['result']:
258                    raise error
259                    logging.debug(unlocked_device.content())
260                else:
261                    self._locked_aps.remove(ap_locker)
262                return
263
264        logging.error('Tried to unlock a host we have not locked (%s)?',
265                      host_name)
266
267
268    def unlock_aps(self):
269        """Unlock APs after we're done."""
270        # Make a copy of all of the hostnames to process
271        host_names = list()
272        for ap_locker in self._locked_aps:
273            host_names.append(ap_locker.configurator.host_name)
274        for host_name in host_names:
275            # TODO(@rjahagir): Change method to datastore.
276            # self.unlock_one_ap_in_datastore(host_name)
277            self.unlock_one_ap(host_name)
278
279
280    def unlock_and_reclaim_ap(self, host_name):
281        """Unlock an AP but return it to the remaining batch of APs.
282
283        @param host_name: a string, host name.
284        """
285        for ap_locker in self._locked_aps:
286            if host_name == ap_locker.configurator.host_name:
287                self.aps_to_lock.append(ap_locker)
288                # TODO(@rjahagir): Change method to datastore.
289                # self.unlock_one_ap_in_datastore(host_name)
290                self.unlock_one_ap(host_name)
291                return
292
293
294    def unlock_and_reclaim_aps(self):
295        """Unlock APs but return them to the batch of remining APs.
296
297        unlock_aps() will remove the remaining APs from the list of all APs
298        to process.  This method will add the remaining APs back to the pool
299        of unprocessed APs.
300
301        """
302        # Add the APs back into the pool
303        self.aps_to_lock.extend(self._locked_aps)
304        self.unlock_aps()
305