##// END OF EJS Templates
build(dev): synced debug deps with -ce edition
build(dev): synced debug deps with -ce edition

File last commit:

r1152:a0c49580 default
r1183:63fb9daf default
Show More
test_subprocessio.py
155 lines | 4.8 KiB | text/x-python | PythonLexer
/ vcsserver / tests / test_subprocessio.py
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 # RhodeCode VCSServer provides access to different vcs backends via network.
source-code: updated copyrights to 2023
r1126 # Copyright (C) 2014-2023 RhodeCode GmbH
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 #
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software Foundation,
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
import io
import os
import sys
import pytest
from vcsserver import subprocessio
packages: move the str utils to it's own module
r1060 from vcsserver.str_utils import ascii_bytes
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
lint: auto-fixes
r1152 class FileLikeObj: # pragma: no cover
tests: reduced memory footprint on subprocession tests....
r638
python3: code change for py3 support...
r1048 def __init__(self, data: bytes, size):
chunks = size // len(data)
tests: reduced memory footprint on subprocession tests....
r638
self.stream = self._get_stream(data, chunks)
def _get_stream(self, data, chunks):
py3: drop xrange
r982 for x in range(chunks):
tests: reduced memory footprint on subprocession tests....
r638 yield data
def read(self, n):
python3: code change for py3 support...
r1048 buffer_stream = b''
tests: reduced memory footprint on subprocession tests....
r638 for chunk in self.stream:
buffer_stream += chunk
if len(buffer_stream) >= n:
break
# self.stream = self.bytes[n:]
return buffer_stream
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 @pytest.fixture(scope='module')
def environ():
"""Delete coverage variables, as they make the tests fail."""
env = dict(os.environ)
code: flake8 fixes
r1063 for key in list(env.keys()):
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 if key.startswith('COV_CORE_'):
del env[key]
return env
def _get_python_args(script):
code: format style and py3 compatability
r589 return [sys.executable, '-c', 'import sys; import time; import shutil; ' + script]
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
def test_raise_exception_on_non_zero_return_code(environ):
python3: code change for py3 support...
r1048 call_args = _get_python_args('raise ValueError("fail")')
with pytest.raises(OSError):
b''.join(subprocessio.SubprocessIOChunker(call_args, shell=False, env=environ))
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
def test_does_not_fail_on_non_zero_return_code(environ):
python3: code change for py3 support...
r1048 call_args = _get_python_args('sys.stdout.write("hello"); sys.exit(1)')
proc = subprocessio.SubprocessIOChunker(call_args, shell=False, fail_on_return_code=False, env=environ)
output = b''.join(proc)
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
python3: code change for py3 support...
r1048 assert output == b'hello'
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
def test_raise_exception_on_stderr(environ):
python3: code change for py3 support...
r1048 call_args = _get_python_args('sys.stderr.write("WRITE_TO_STDERR"); time.sleep(1);')
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
python3: code change for py3 support...
r1048 with pytest.raises(OSError) as excinfo:
b''.join(subprocessio.SubprocessIOChunker(call_args, shell=False, env=environ))
assert 'exited due to an error:\nWRITE_TO_STDERR' in str(excinfo.value)
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
def test_does_not_fail_on_stderr(environ):
python3: code change for py3 support...
r1048 call_args = _get_python_args('sys.stderr.write("WRITE_TO_STDERR"); sys.stderr.flush; time.sleep(2);')
proc = subprocessio.SubprocessIOChunker(call_args, shell=False, fail_on_stderr=False, env=environ)
output = b''.join(proc)
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
python3: code change for py3 support...
r1048 assert output == b''
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
python3: code change for py3 support...
r1048 @pytest.mark.parametrize('size', [
1,
10 ** 5
])
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 def test_output_with_no_input(size, environ):
python3: code change for py3 support...
r1048 call_args = _get_python_args(f'sys.stdout.write("X" * {size});')
proc = subprocessio.SubprocessIOChunker(call_args, shell=False, env=environ)
output = b''.join(proc)
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
python3: code change for py3 support...
r1048 assert output == ascii_bytes("X" * size)
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
python3: code change for py3 support...
r1048 @pytest.mark.parametrize('size', [
1,
10 ** 5
])
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 def test_output_with_no_input_does_not_fail(size, environ):
python3: code change for py3 support...
r1048 call_args = _get_python_args(f'sys.stdout.write("X" * {size}); sys.exit(1)')
proc = subprocessio.SubprocessIOChunker(call_args, shell=False, fail_on_return_code=False, env=environ)
output = b''.join(proc)
assert output == ascii_bytes("X" * size)
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
python3: code change for py3 support...
r1048 @pytest.mark.parametrize('size', [
1,
10 ** 5
])
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 def test_output_with_input(size, environ):
tests: reduced memory footprint on subprocession tests....
r638 data_len = size
python3: code change for py3 support...
r1048 inputstream = FileLikeObj(b'X', size)
tests: reduced memory footprint on subprocession tests....
r638
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 # This acts like the cat command.
python3: code change for py3 support...
r1048 call_args = _get_python_args('shutil.copyfileobj(sys.stdin, sys.stdout)')
# note: in this tests we explicitly don't assign chunker to a variable and let it stream directly
output = b''.join(
subprocessio.SubprocessIOChunker(call_args, shell=False, input_stream=inputstream, env=environ)
code: format style and py3 compatability
r589 )
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
tests: reduced memory footprint on subprocession tests....
r638 assert len(output) == data_len
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
python3: code change for py3 support...
r1048 @pytest.mark.parametrize('size', [
1,
10 ** 5
])
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 def test_output_with_input_skipping_iterator(size, environ):
tests: reduced memory footprint on subprocession tests....
r638 data_len = size
python3: code change for py3 support...
r1048 inputstream = FileLikeObj(b'X', size)
tests: reduced memory footprint on subprocession tests....
r638
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130 # This acts like the cat command.
python3: code change for py3 support...
r1048 call_args = _get_python_args('shutil.copyfileobj(sys.stdin, sys.stdout)')
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
# Note: assigning the chunker makes sure that it is not deleted too early
python3: code change for py3 support...
r1048 proc = subprocessio.SubprocessIOChunker(call_args, shell=False, input_stream=inputstream, env=environ)
output = b''.join(proc.stdout)
packaging: moved tests into the main library itself. This is consistent with how our other projects do it.
r130
tests: reduced memory footprint on subprocession tests....
r638 assert len(output) == data_len