Derek Beckett | 63e1c44 | 2020-08-11 14:49:47 -0700 | [diff] [blame] | 1 | # Lint as: python2, python3 |
| 2 | from __future__ import absolute_import |
| 3 | from __future__ import division |
| 4 | from __future__ import print_function |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 5 | import itertools |
jadmanski | 91cd58c | 2008-12-02 15:34:53 +0000 | [diff] [blame] | 6 | import common |
Derek Beckett | 63e1c44 | 2020-08-11 14:49:47 -0700 | [diff] [blame] | 7 | import six |
jadmanski | 91cd58c | 2008-12-02 15:34:53 +0000 | [diff] [blame] | 8 | |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 9 | |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 10 | def _get_unpassable_types(arg): |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 11 | """ Given an argument, returns a set of types contained in arg that are |
| 12 | unpassable. If arg is an atomic type (e.g. int) it either returns an |
| 13 | empty set (if the type is passable) or a singleton of the type (if the |
| 14 | type is not passable). """ |
Derek Beckett | 63e1c44 | 2020-08-11 14:49:47 -0700 | [diff] [blame] | 15 | if isinstance(arg, (six.string_types, int, int)): |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 16 | return set() |
| 17 | elif isinstance(arg, (list, tuple, set, frozenset, dict)): |
| 18 | if isinstance(arg, dict): |
| 19 | # keys and values must both be passable |
Derek Beckett | 63e1c44 | 2020-08-11 14:49:47 -0700 | [diff] [blame] | 20 | parts = itertools.chain(six.iterkeys(arg), six.itervalues(arg)) |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 21 | else: |
| 22 | # for all other containers we just iterate |
| 23 | parts = iter(arg) |
| 24 | types = set() |
| 25 | for part in parts: |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 26 | types |= _get_unpassable_types(part) |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 27 | return types |
| 28 | else: |
| 29 | return set([type(arg)]) |
| 30 | |
| 31 | |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 32 | def _validate_args(args): |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 33 | """ Validates arguments. Lists and dictionaries are valid argument types, |
| 34 | so you can pass *args and **dargs in directly, rather than having to |
| 35 | iterate over them yourself. """ |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 36 | unpassable_types = _get_unpassable_types(args) |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 37 | if unpassable_types: |
| 38 | msg = "arguments of type '%s' cannot be passed to remote profilers" |
| 39 | msg %= ", ".join(t.__name__ for t in unpassable_types) |
| 40 | raise TypeError(msg) |
| 41 | |
| 42 | |
| 43 | class profiler_proxy(object): |
| 44 | """ This is a server-side class that acts as a proxy to a real client-side |
| 45 | profiler class.""" |
| 46 | |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 47 | def __init__(self, profiler_name): |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 48 | self.name = profiler_name |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 49 | |
jadmanski | 91cd58c | 2008-12-02 15:34:53 +0000 | [diff] [blame] | 50 | # does the profiler support rebooting? |
| 51 | profiler_module = common.setup_modules.import_module( |
| 52 | profiler_name, "autotest_lib.client.profilers.%s" % profiler_name) |
| 53 | profiler_class = getattr(profiler_module, profiler_name) |
| 54 | self.supports_reboot = profiler_class.supports_reboot |
| 55 | |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 56 | |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 57 | def initialize(self, *args, **dargs): |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 58 | _validate_args(args) |
| 59 | _validate_args(dargs) |
mbligh | b3c0c91 | 2008-11-27 00:32:45 +0000 | [diff] [blame] | 60 | self.args, self.dargs = args, dargs |
| 61 | |
| 62 | |
| 63 | def setup(self, *args, **dargs): |
| 64 | assert self.args == args and self.dargs == dargs |
| 65 | # the actual setup happens lazily at start() |
| 66 | |
| 67 | |
jadmanski | 4f90925 | 2008-12-01 20:47:10 +0000 | [diff] [blame] | 68 | def start(self, test, host=None): |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 69 | raise NotImplementedError('start not implemented') |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 70 | |
| 71 | |
jadmanski | 4f90925 | 2008-12-01 20:47:10 +0000 | [diff] [blame] | 72 | def stop(self, test, host=None): |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 73 | raise NotImplementedError('stop not implemented') |
jadmanski | 043e113 | 2008-11-19 17:10:32 +0000 | [diff] [blame] | 74 | |
| 75 | |
jadmanski | 4f90925 | 2008-12-01 20:47:10 +0000 | [diff] [blame] | 76 | def report(self, test, host=None, wait_on_client=True): |
mbligh | 15971eb | 2009-12-29 02:55:23 +0000 | [diff] [blame] | 77 | raise NotImplementedError('report not implemented') |