#!/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()