| #!/usr/bin/env python3 |
| |
| # Copyright 2016 - The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| # Import the python3 compatible bytes() |
| from builtins import bytes |
| |
| import mock |
| import os |
| import sys |
| import unittest |
| |
| from acts.libs.proc import job |
| |
| if os.name == 'posix' and sys.version_info[0] < 3: |
| import subprocess32 as subprocess |
| else: |
| import subprocess |
| |
| |
| class FakePopen(object): |
| """A fake version of the object returned from subprocess.Popen().""" |
| |
| def __init__(self, |
| stdout=None, |
| stderr=None, |
| returncode=0, |
| will_timeout=False): |
| self.returncode = returncode |
| self._stdout = bytes(stdout, |
| 'utf-8') if stdout is not None else bytes() |
| self._stderr = bytes(stderr, |
| 'utf-8') if stderr is not None else bytes() |
| self._will_timeout = will_timeout |
| |
| def communicate(self, timeout=None): |
| if self._will_timeout: |
| raise subprocess.TimeoutExpired( |
| -1, 'Timed out according to test logic') |
| return self._stdout, self._stderr |
| |
| def kill(self): |
| pass |
| |
| def wait(self): |
| pass |
| |
| |
| class JobTestCases(unittest.TestCase): |
| @mock.patch( |
| 'acts.libs.proc.job.subprocess.Popen', |
| return_value=FakePopen(stdout='TEST\n')) |
| def test_run_success(self, popen): |
| """Test running a simple shell command.""" |
| result = job.run('echo TEST') |
| self.assertTrue(result.stdout.startswith('TEST')) |
| |
| @mock.patch( |
| 'acts.libs.proc.job.subprocess.Popen', |
| return_value=FakePopen(stderr='TEST\n')) |
| def test_run_stderr(self, popen): |
| """Test that we can read process stderr.""" |
| result = job.run('echo TEST 1>&2') |
| self.assertEqual(len(result.stdout), 0) |
| self.assertTrue(result.stderr.startswith('TEST')) |
| self.assertFalse(result.stdout) |
| |
| @mock.patch( |
| 'acts.libs.proc.job.subprocess.Popen', |
| return_value=FakePopen(returncode=1)) |
| def test_run_error(self, popen): |
| """Test that we raise on non-zero exit statuses.""" |
| self.assertRaises(job.Error, job.run, 'exit 1') |
| |
| @mock.patch( |
| 'acts.libs.proc.job.subprocess.Popen', |
| return_value=FakePopen(returncode=1)) |
| def test_run_with_ignored_error(self, popen): |
| """Test that we can ignore exit status on request.""" |
| result = job.run('exit 1', ignore_status=True) |
| self.assertEqual(result.exit_status, 1) |
| |
| @mock.patch( |
| 'acts.libs.proc.job.subprocess.Popen', |
| return_value=FakePopen(will_timeout=True)) |
| def test_run_timeout(self, popen): |
| """Test that we correctly implement command timeouts.""" |
| self.assertRaises(job.Error, job.run, 'sleep 5', timeout=0.1) |
| |
| @mock.patch( |
| 'acts.libs.proc.job.subprocess.Popen', |
| return_value=FakePopen(stdout='TEST\n')) |
| def test_run_no_shell(self, popen): |
| """Test that we handle running without a wrapping shell.""" |
| result = job.run(['echo', 'TEST']) |
| self.assertTrue(result.stdout.startswith('TEST')) |
| |
| @mock.patch( |
| 'acts.libs.proc.job.subprocess.Popen', |
| return_value=FakePopen(stdout='TEST\n')) |
| def test_job_env(self, popen): |
| """Test that we can set environment variables correctly.""" |
| test_env = {'MYTESTVAR': '20'} |
| result = job.run('printenv', env=test_env.copy()) |
| popen.assert_called_once() |
| _, kwargs = popen.call_args |
| self.assertTrue('env' in kwargs) |
| self.assertEqual(kwargs['env'], test_env) |
| |
| |
| if __name__ == '__main__': |
| unittest.main() |