1# Copyright 2015 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
5import collections
6import time
7
8from autotest_lib.client.common_lib import error
9from autotest_lib.client.common_lib import utils
10
11# Use this with ProcessWatcher to start your process in a minijail.  This
12# is useful for instance if you would like to drop autotest's default root
13# priviledges.  Both fields must be set to valid users/groups.
14MinijailConfig = collections.namedtuple('MinijailConfig', ['user', 'group'])
15
16
17class ProcessWatcher(object):
18    """Start a process, and terminate it later."""
19
20    def __init__(self, command, args=tuple(), minijail_config=None, host=None):
21        """Construst a ProcessWatcher without starting the process.
22
23        @param command: string command to use to start the process.
24        @param args: list of strings to pass to the command.
25        @param minijail_config: MinijailConfig tuple defined above.
26        @param host: host object if the server should be started on a remote
27                host.
28
29        """
30        self._command = ' '.join([command] + list(args))
31        if '"' in self._command:
32            raise error.TestError('Please implement shell escaping in '
33                                  'ProcessWatcher.')
34        self._minijail_config = minijail_config
35        self._run = utils.run if host is None else host.run
36
37
38    def start(self):
39        """Start a (potentially remote) instance of the process."""
40        command = self._command
41        prefix = ''
42        if self._minijail_config is not None:
43            prefix = 'minijail0 -i -g %s -u %s ' % (self._minijail_config.group,
44                                                    self._minijail_config.user)
45        # Redirect output streams to avoid odd interactions between autotest's
46        # shell environment and the command's runtime environment.
47        self._run('%s%s >/dev/null 2>&1 &' % (prefix, self._command))
48
49
50    def close(self, timeout_seconds=40):
51        """Close the (potentially remote) instance of the process.
52
53        @param timeout_seconds: int number of seconds to wait for shutdown.
54
55        """
56        self._run('pkill -f --signal TERM "%s"' % self._command,
57                  ignore_status=True)
58        start_time = time.time()
59        while time.time() - start_time < timeout_seconds:
60            result = self._run('pgrep -f -l "%s"' % self._command,
61                               ignore_status=True)
62            if result.exit_status != 0:
63                return
64            time.sleep(0.3)
65        raise error.TestError('Timed out waiting for %s to die.' %
66                              self._command)
67