123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851 |
- #
- # The Qubes OS Project, https://www.qubes-os.org/
- #
- # Copyright (C) 2014-2020
- # Marek Marczykowski-Górecki <marmarek@invisiblethingslab.com>
- # Copyright (C) 2015 Wojtek Porczyk <woju@invisiblethingslab.com>
- #
- # This library is free software; you can redistribute it and/or
- # modify it under the terms of the GNU Lesser General Public
- # License as published by the Free Software Foundation; either
- # version 2.1 of the License, or (at your option) any later version.
- #
- # This library 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
- # Lesser General Public License for more details.
- #
- # You should have received a copy of the GNU Lesser General Public
- # License along with this library; if not, see <https://www.gnu.org/licenses/>.
- #
- import asyncio
- import contextlib
- import os
- import subprocess
- import sys
- import qubes.config
- import qubes.devices
- import qubes.tests
- import qubes.vm.appvm
- import qubes.vm.templatevm
- TEST_DATA = b"0123456789" * 1024
- class TC_00_QrexecMixin(object):
- def setUp(self):
- super().setUp()
- self.init_default_template(self.template)
- if self._testMethodName == 'test_210_time_sync':
- self.init_networking()
- self.testvm1 = self.app.add_new_vm(
- qubes.vm.appvm.AppVM,
- label='red',
- name=self.make_vm_name('vm1'),
- template=self.app.domains[self.template])
- self.loop.run_until_complete(self.testvm1.create_on_disk())
- self.testvm2 = self.app.add_new_vm(
- qubes.vm.appvm.AppVM,
- label='red',
- name=self.make_vm_name('vm2'),
- template=self.app.domains[self.template])
- self.loop.run_until_complete(self.testvm2.create_on_disk())
- self.app.save()
- def tearDown(self):
- # socket-based qrexec tests:
- if os.path.exists('/etc/qubes-rpc/test.Socket'):
- os.unlink('/etc/qubes-rpc/test.Socket')
- if hasattr(self, 'service_proc'):
- try:
- self.service_proc.terminate()
- self.loop.run_until_complete(self.service_proc.communicate())
- except ProcessLookupError:
- pass
- super().tearDown()
- def test_050_qrexec_simple_eof(self):
- """Test for data and EOF transmission dom0->VM"""
- # XXX is this still correct? this is no longer simple qrexec,
- # but qubes.VMShell
- self.loop.run_until_complete(self.testvm1.start())
- try:
- (stdout, stderr) = self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_for_stdio('cat', input=TEST_DATA),
- timeout=10))
- except asyncio.TimeoutError:
- self.fail(
- "Timeout, probably EOF wasn't transferred to the VM process")
- self.assertEqual(stdout, TEST_DATA,
- 'Received data differs from what was sent')
- self.assertFalse(stderr,
- 'Some data was printed to stderr')
- def test_051_qrexec_simple_eof_reverse(self):
- """Test for EOF transmission VM->dom0"""
- @asyncio.coroutine
- def run(self):
- p = yield from self.testvm1.run(
- 'echo test; exec >&-; cat > /dev/null',
- stdin=subprocess.PIPE,
- stdout=subprocess.PIPE,
- stderr=subprocess.PIPE)
- # this will hang on test failure
- stdout = yield from asyncio.wait_for(p.stdout.read(), timeout=10)
- p.stdin.write(TEST_DATA)
- yield from p.stdin.drain()
- p.stdin.close()
- self.assertEqual(stdout.strip(), b'test',
- 'Received data differs from what was expected')
- # this may hang in some buggy cases
- self.assertFalse((yield from p.stderr.read()),
- 'Some data was printed to stderr')
- try:
- yield from asyncio.wait_for(p.wait(), timeout=1)
- except asyncio.TimeoutError:
- self.fail("Timeout, "
- "probably EOF wasn't transferred from the VM process")
- self.loop.run_until_complete(self.testvm1.start())
- self.loop.run_until_complete(self.wait_for_session(self.testvm1))
- self.loop.run_until_complete(run(self))
- def test_052_qrexec_vm_service_eof(self):
- """Test for EOF transmission VM(src)->VM(dst)"""
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- self.loop.run_until_complete(asyncio.wait([
- self.wait_for_session(self.testvm1),
- self.wait_for_session(self.testvm2)]))
- self.create_remote_file(self.testvm2,
- '/etc/qubes-rpc/test.EOF',
- '#!/bin/sh\n/bin/cat\n')
- with self.qrexec_policy('test.EOF', self.testvm1, self.testvm2):
- try:
- stdout, _ = self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_for_stdio('''\
- /usr/lib/qubes/qrexec-client-vm {} test.EOF \
- /bin/sh -c 'echo test; exec >&-; cat >&$SAVED_FD_1'
- '''.format(self.testvm2.name)),
- timeout=10))
- except subprocess.CalledProcessError as e:
- self.fail('{} exited with non-zero code {}; stderr: {}'.format(
- e.cmd, e.returncode, e.stderr))
- except asyncio.TimeoutError:
- self.fail("Timeout, probably EOF wasn't transferred")
- self.assertEqual(stdout, b'test\n',
- 'Received data differs from what was expected')
- def test_053_qrexec_vm_service_eof_reverse(self):
- """Test for EOF transmission VM(src)<-VM(dst)"""
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.EOF',
- '#!/bin/sh\n'
- 'echo test; exec >&-; cat >/dev/null')
- with self.qrexec_policy('test.EOF', self.testvm1, self.testvm2):
- try:
- stdout, _ = self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_for_stdio('''\
- /usr/lib/qubes/qrexec-client-vm {} test.EOF \
- /bin/sh -c 'cat >&$SAVED_FD_1'
- '''.format(self.testvm2.name)),
- timeout=10))
- except subprocess.CalledProcessError as e:
- self.fail('{} exited with non-zero code {}; stderr: {}'.format(
- e.cmd, e.returncode, e.stderr))
- except asyncio.TimeoutError:
- self.fail("Timeout, probably EOF wasn't transferred")
- self.assertEqual(stdout, b'test\n',
- 'Received data differs from what was expected')
- def test_055_qrexec_dom0_service_abort(self):
- """
- Test if service abort (by dom0) is properly handled by source VM.
- If "remote" part of the service terminates, the source part should
- properly be notified. This includes closing its stdin (which is
- already checked by test_053_qrexec_vm_service_eof_reverse), but also
- its stdout - otherwise such service might hang on write(2) call.
- """
- self.loop.run_until_complete(self.testvm1.start())
- self.create_local_file('/etc/qubes-rpc/test.Abort',
- 'sleep 1')
- with self.qrexec_policy('test.Abort', self.testvm1, 'dom0'):
- try:
- # two possible exit codes, depending on when exactly dom0
- # service terminates:
- # exit code 141: EPIPE (no buffered data)
- # exit code 1: ECONNRESET (some buffered data remains)
- stdout, _ = self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_for_stdio('''\
- /usr/lib/qubes/qrexec-client-vm dom0 test.Abort \
- /bin/sh -c 'cat /dev/zero; echo $? >/tmp/exit-code';
- sleep 1;
- e=$(cat /tmp/exit-code);
- test $e -eq 141 -o $e -eq 1'''),
- timeout=10))
- except subprocess.CalledProcessError as e:
- self.fail('{} exited with non-zero code {}; stderr: {}'.format(
- e.cmd, e.returncode, e.stderr))
- except asyncio.TimeoutError:
- self.fail("Timeout, probably stdout wasn't closed")
- def test_060_qrexec_exit_code_dom0(self):
- self.loop.run_until_complete(self.testvm1.start())
- self.loop.run_until_complete(self.testvm1.run_for_stdio('exit 0'))
- with self.assertRaises(subprocess.CalledProcessError) as e:
- self.loop.run_until_complete(self.testvm1.run_for_stdio('exit 3'))
- self.assertEqual(e.exception.returncode, 3)
- def test_065_qrexec_exit_code_vm(self):
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- with self.qrexec_policy('test.Retcode', self.testvm1, self.testvm2):
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.Retcode',
- 'exit 0')
- (stdout, stderr) = self.loop.run_until_complete(
- self.testvm1.run_for_stdio('''\
- /usr/lib/qubes/qrexec-client-vm {} test.Retcode;
- echo $?'''.format(self.testvm2.name),
- stderr=None))
- self.assertEqual(stdout, b'0\n')
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.Retcode',
- 'exit 3')
- (stdout, stderr) = self.loop.run_until_complete(
- self.testvm1.run_for_stdio('''\
- /usr/lib/qubes/qrexec-client-vm {} test.Retcode;
- echo $?'''.format(self.testvm2.name),
- stderr=None))
- self.assertEqual(stdout, b'3\n')
- def test_070_qrexec_vm_simultaneous_write(self):
- """Test for simultaneous write in VM(src)->VM(dst) connection
- This is regression test for #1347
- Check for deadlock when initially both sides writes a lot of data
- (and not read anything). When one side starts reading, it should
- get the data and the remote side should be possible to write then more.
- There was a bug where remote side was waiting on write(2) and not
- handling anything else.
- """
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.write', '''\
- # first write a lot of data
- dd if=/dev/zero bs=993 count=10000 iflag=fullblock
- # and only then read something
- dd of=/dev/null bs=993 count=10000 iflag=fullblock
- ''')
- with self.qrexec_policy('test.write', self.testvm1, self.testvm2):
- try:
- self.loop.run_until_complete(asyncio.wait_for(
- # first write a lot of data to fill all the buffers
- # then after some time start reading
- self.testvm1.run_for_stdio('''\
- /usr/lib/qubes/qrexec-client-vm {} test.write \
- /bin/sh -c '
- dd if=/dev/zero bs=993 count=10000 iflag=fullblock &
- sleep 1;
- dd of=/dev/null bs=993 count=10000 iflag=fullblock;
- wait'
- '''.format(self.testvm2.name)), timeout=10))
- except subprocess.CalledProcessError as e:
- self.fail('{} exited with non-zero code {}; stderr: {}'.format(
- e.cmd, e.returncode, e.stderr))
- except asyncio.TimeoutError:
- self.fail('Timeout, probably deadlock')
- def test_071_qrexec_dom0_simultaneous_write(self):
- """Test for simultaneous write in dom0(src)->VM(dst) connection
- Similar to test_070_qrexec_vm_simultaneous_write, but with dom0
- as a source.
- """
- self.loop.run_until_complete(self.testvm2.start())
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.write', '''\
- # first write a lot of data
- dd if=/dev/zero bs=993 count=10000 iflag=fullblock
- # and only then read something
- dd of=/dev/null bs=993 count=10000 iflag=fullblock
- ''')
- # can't use subprocess.PIPE, because asyncio will claim those FDs
- pipe1_r, pipe1_w = os.pipe()
- pipe2_r, pipe2_w = os.pipe()
- try:
- local_proc = self.loop.run_until_complete(
- asyncio.create_subprocess_shell(
- # first write a lot of data to fill all the buffers
- "dd if=/dev/zero bs=993 count=10000 iflag=fullblock & "
- # then after some time start reading
- "sleep 1; "
- "dd of=/dev/null bs=993 count=10000 iflag=fullblock; "
- "wait", stdin=pipe1_r, stdout=pipe2_w))
- self.service_proc = self.loop.run_until_complete(self.testvm2.run_service(
- "test.write", stdin=pipe2_r, stdout=pipe1_w))
- finally:
- os.close(pipe1_r)
- os.close(pipe1_w)
- os.close(pipe2_r)
- os.close(pipe2_w)
- try:
- self.loop.run_until_complete(
- asyncio.wait_for(self.service_proc.wait(), timeout=10))
- except asyncio.TimeoutError:
- self.fail("Timeout, probably deadlock")
- else:
- self.assertEqual(self.service_proc.returncode, 0,
- "Service call failed")
- def test_072_qrexec_to_dom0_simultaneous_write(self):
- """Test for simultaneous write in dom0(src)<-VM(dst) connection
- Similar to test_071_qrexec_dom0_simultaneous_write, but with dom0
- as a "hanging" side.
- """
- self.loop.run_until_complete(self.testvm2.start())
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.write', '''\
- # first write a lot of data
- dd if=/dev/zero bs=993 count=10000 iflag=fullblock &
- # and only then read something
- dd of=/dev/null bs=993 count=10000 iflag=fullblock
- sleep 1;
- wait
- ''')
- # can't use subprocess.PIPE, because asyncio will claim those FDs
- pipe1_r, pipe1_w = os.pipe()
- pipe2_r, pipe2_w = os.pipe()
- try:
- local_proc = self.loop.run_until_complete(
- asyncio.create_subprocess_shell(
- # first write a lot of data to fill all the buffers
- "dd if=/dev/zero bs=993 count=10000 iflag=fullblock & "
- # then, only when all written, read something
- "dd of=/dev/null bs=993 count=10000 iflag=fullblock; ",
- stdin=pipe1_r, stdout=pipe2_w))
- self.service_proc = self.loop.run_until_complete(
- self.testvm2.run_service(
- "test.write", stdin=pipe2_r, stdout=pipe1_w))
- finally:
- os.close(pipe1_r)
- os.close(pipe1_w)
- os.close(pipe2_r)
- os.close(pipe2_w)
- try:
- self.loop.run_until_complete(
- asyncio.wait_for(self.service_proc.wait(), timeout=10))
- except asyncio.TimeoutError:
- self.fail("Timeout, probably deadlock")
- else:
- self.assertEqual(self.service_proc.returncode, 0,
- "Service call failed")
- def test_080_qrexec_service_argument_allow_default(self):
- """Qrexec service call with argument"""
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.Argument',
- '/usr/bin/printf %s "$1"')
- with self.qrexec_policy('test.Argument', self.testvm1, self.testvm2):
- stdout, stderr = self.loop.run_until_complete(
- self.testvm1.run_for_stdio('/usr/lib/qubes/qrexec-client-vm '
- '{} test.Argument+argument'.format(self.testvm2.name),
- stderr=None))
- self.assertEqual(stdout, b'argument')
- def test_081_qrexec_service_argument_allow_specific(self):
- """Qrexec service call with argument - allow only specific value"""
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.Argument',
- '/usr/bin/printf %s "$1"')
- with self.qrexec_policy('test.Argument', '$anyvm', '$anyvm', False):
- with self.qrexec_policy('test.Argument+argument',
- self.testvm1.name, self.testvm2.name):
- stdout, stderr = self.loop.run_until_complete(
- self.testvm1.run_for_stdio(
- '/usr/lib/qubes/qrexec-client-vm '
- '{} test.Argument+argument'.format(self.testvm2.name),
- stderr=None))
- self.assertEqual(stdout, b'argument')
- def test_082_qrexec_service_argument_deny_specific(self):
- """Qrexec service call with argument - deny specific value"""
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.Argument',
- '/usr/bin/printf %s "$1"')
- with self.qrexec_policy('test.Argument', '$anyvm', '$anyvm'):
- with self.qrexec_policy('test.Argument+argument',
- self.testvm1, self.testvm2, allow=False):
- with self.assertRaises(subprocess.CalledProcessError,
- msg='Service request should be denied'):
- self.loop.run_until_complete(
- self.testvm1.run_for_stdio(
- '/usr/lib/qubes/qrexec-client-vm {} '
- 'test.Argument+argument'.format(self.testvm2.name),
- stderr=None))
- def test_083_qrexec_service_argument_specific_implementation(self):
- """Qrexec service call with argument - argument specific
- implementatation"""
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- self.create_remote_file(self.testvm2,
- '/etc/qubes-rpc/test.Argument',
- '/usr/bin/printf %s "$1"')
- self.create_remote_file(self.testvm2,
- '/etc/qubes-rpc/test.Argument+argument',
- '/usr/bin/printf "specific: %s" "$1"')
- with self.qrexec_policy('test.Argument', self.testvm1, self.testvm2):
- stdout, stderr = self.loop.run_until_complete(
- self.testvm1.run_for_stdio('/usr/lib/qubes/qrexec-client-vm '
- '{} test.Argument+argument'.format(self.testvm2.name),
- stderr=None))
- self.assertEqual(stdout, b'specific: argument')
- def test_084_qrexec_service_argument_extra_env(self):
- """Qrexec service call with argument - extra env variables"""
- self.loop.run_until_complete(asyncio.wait([
- self.testvm1.start(),
- self.testvm2.start()]))
- self.create_remote_file(self.testvm2, '/etc/qubes-rpc/test.Argument',
- '/usr/bin/printf "%s %s" '
- '"$QREXEC_SERVICE_FULL_NAME" "$QREXEC_SERVICE_ARGUMENT"')
- with self.qrexec_policy('test.Argument', self.testvm1, self.testvm2):
- stdout, stderr = self.loop.run_until_complete(
- self.testvm1.run_for_stdio('/usr/lib/qubes/qrexec-client-vm '
- '{} test.Argument+argument'.format(self.testvm2.name),
- stderr=None))
- self.assertEqual(stdout, b'test.Argument+argument argument')
- def test_090_qrexec_service_socket_dom0(self):
- """Basic test socket services (dom0) - data receive"""
- self.loop.run_until_complete(self.testvm1.start())
- self.service_proc = self.loop.run_until_complete(
- asyncio.create_subprocess_shell(
- 'socat -u UNIX-LISTEN:/etc/qubes-rpc/test.Socket,mode=666 -',
- stdout=subprocess.PIPE, stdin=subprocess.PIPE))
- try:
- with self.qrexec_policy('test.Socket', self.testvm1, '@adminvm'):
- (stdout, stderr) = self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_for_stdio(
- 'qrexec-client-vm @adminvm test.Socket', input=TEST_DATA),
- timeout=10))
- except subprocess.CalledProcessError as e:
- self.fail('{} exited with non-zero code {}; stderr: {}'.format(
- e.cmd, e.returncode, e.stderr))
- except asyncio.TimeoutError:
- self.fail(
- "service timeout, probably EOF wasn't transferred to the VM process")
- try:
- (service_stdout, service_stderr) = self.loop.run_until_complete(
- asyncio.wait_for(
- self.service_proc.communicate(),
- timeout=10))
- except asyncio.TimeoutError:
- self.fail(
- "socat timeout, probably EOF wasn't transferred to the VM process")
- service_descriptor = b'test.Socket+ test-inst-vm1 keyword adminvm\0'
- self.assertEqual(service_stdout, service_descriptor + TEST_DATA,
- 'Received data differs from what was sent')
- self.assertFalse(stderr,
- 'Some data was printed to stderr')
- self.assertFalse(service_stderr,
- 'Some data was printed to stderr')
- def test_091_qrexec_service_socket_dom0_send(self):
- """Basic test socket services (dom0) - data send"""
- self.loop.run_until_complete(self.testvm1.start())
- self.create_local_file('/tmp/service-input', TEST_DATA.decode())
- self.service_proc = self.loop.run_until_complete(
- asyncio.create_subprocess_shell(
- 'socat -u OPEN:/tmp/service-input UNIX-LISTEN:/etc/qubes-rpc/test.Socket,mode=666'))
- try:
- with self.qrexec_policy('test.Socket', self.testvm1, '@adminvm'):
- stdout, stderr = self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_for_stdio(
- 'qrexec-client-vm @adminvm test.Socket'),
- timeout=10))
- except subprocess.CalledProcessError as e:
- self.fail('{} exited with non-zero code {}; stderr: {}'.format(
- e.cmd, e.returncode, e.stderr))
- except asyncio.TimeoutError:
- self.fail(
- "service timeout, probably EOF wasn't transferred to the VM process")
- try:
- (service_stdout, service_stderr) = self.loop.run_until_complete(
- asyncio.wait_for(
- self.service_proc.communicate(),
- timeout=10))
- except asyncio.TimeoutError:
- self.fail(
- "socat timeout, probably EOF wasn't transferred to the VM process")
- self.assertEqual(stdout, TEST_DATA,
- 'Received data differs from what was sent')
- self.assertFalse(stderr,
- 'Some data was printed to stderr')
- self.assertFalse(service_stderr,
- 'Some data was printed to stderr')
- def test_092_qrexec_service_socket_dom0_eof_reverse(self):
- """Test for EOF transmission dom0(socket)->VM"""
- self.loop.run_until_complete(self.testvm1.start())
- self.create_local_file(
- '/tmp/service_script',
- '#!/usr/bin/python3\n'
- 'import socket, os, sys, time\n'
- 's = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)\n'
- 'os.umask(0)\n'
- 's.bind("/etc/qubes-rpc/test.Socket")\n'
- 's.listen(1)\n'
- 'conn, addr = s.accept()\n'
- 'conn.send(b"test\\n")\n'
- 'conn.shutdown(socket.SHUT_WR)\n'
- # wait longer than the timeout below
- 'time.sleep(15)\n'
- )
- self.service_proc = self.loop.run_until_complete(
- asyncio.create_subprocess_shell('python3 /tmp/service_script',
- stdout=subprocess.PIPE, stdin=subprocess.PIPE))
- try:
- with self.qrexec_policy('test.Socket', self.testvm1, '@adminvm'):
- p = self.loop.run_until_complete(self.testvm1.run(
- 'qrexec-client-vm @adminvm test.Socket',
- stdout=subprocess.PIPE, stdin=subprocess.PIPE))
- stdout = self.loop.run_until_complete(asyncio.wait_for(
- p.stdout.read(),
- timeout=10))
- except asyncio.TimeoutError:
- self.fail(
- "service timeout, probably EOF wasn't transferred from the VM process")
- finally:
- with contextlib.suppress(ProcessLookupError):
- p.terminate()
- self.loop.run_until_complete(p.wait())
- self.assertEqual(stdout, b'test\n',
- 'Received data differs from what was expected')
- def test_093_qrexec_service_socket_dom0_eof(self):
- """Test for EOF transmission VM->dom0(socket)"""
- self.loop.run_until_complete(self.testvm1.start())
- self.create_local_file(
- '/tmp/service_script',
- '#!/usr/bin/python3\n'
- 'import socket, os, sys, time\n'
- 's = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)\n'
- 'os.umask(0)\n'
- 's.bind("/etc/qubes-rpc/test.Socket")\n'
- 's.listen(1)\n'
- 'conn, addr = s.accept()\n'
- 'buf = conn.recv(100)\n'
- 'sys.stdout.buffer.write(buf)\n'
- 'buf = conn.recv(10)\n'
- 'sys.stdout.buffer.write(buf)\n'
- 'sys.stdout.buffer.flush()\n'
- 'os.close(1)\n'
- # wait longer than the timeout below
- 'time.sleep(15)\n'
- )
- self.service_proc = self.loop.run_until_complete(
- asyncio.create_subprocess_shell('python3 /tmp/service_script',
- stdout=subprocess.PIPE, stdin=subprocess.PIPE))
- try:
- with self.qrexec_policy('test.Socket', self.testvm1, '@adminvm'):
- p = self.loop.run_until_complete(self.testvm1.run(
- 'qrexec-client-vm @adminvm test.Socket',
- stdin=subprocess.PIPE))
- p.stdin.write(b'test1test2')
- p.stdin.write_eof()
- service_stdout = self.loop.run_until_complete(asyncio.wait_for(
- self.service_proc.stdout.read(),
- timeout=10))
- except asyncio.TimeoutError:
- self.fail(
- "service timeout, probably EOF wasn't transferred from the VM process")
- finally:
- with contextlib.suppress(ProcessLookupError):
- p.terminate()
- self.loop.run_until_complete(p.wait())
- service_descriptor = b'test.Socket+ test-inst-vm1 keyword adminvm\0'
- self.assertEqual(service_stdout, service_descriptor + b'test1test2',
- 'Received data differs from what was expected')
- def _wait_for_socket_setup(self):
- try:
- self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_for_stdio(
- 'while ! test -e /etc/qubes-rpc/test.Socket; do sleep 0.1; done'),
- timeout=10))
- except asyncio.TimeoutError:
- self.fail(
- "waiting for /etc/qubes-rpc/test.Socket in VM timed out")
- def test_095_qrexec_service_socket_vm(self):
- """Basic test socket services (VM) - receive"""
- self.loop.run_until_complete(self.testvm1.start())
- self.service_proc = self.loop.run_until_complete(self.testvm1.run(
- 'socat -u UNIX-LISTEN:/etc/qubes-rpc/test.Socket,mode=666 -',
- stdout=subprocess.PIPE, stdin=subprocess.PIPE,
- user='root'))
- self._wait_for_socket_setup()
- try:
- (stdout, stderr) = self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_service_for_stdio('test.Socket+', input=TEST_DATA),
- timeout=10))
- except subprocess.CalledProcessError as e:
- self.fail('{} exited with non-zero code {}; stderr: {}'.format(
- e.cmd, e.returncode, e.stderr))
- except asyncio.TimeoutError:
- self.fail(
- "service timeout, probably EOF wasn't transferred to the VM process")
- try:
- (service_stdout, service_stderr) = self.loop.run_until_complete(
- asyncio.wait_for(
- self.service_proc.communicate(),
- timeout=10))
- except asyncio.TimeoutError:
- self.fail(
- "socat timeout, probably EOF wasn't transferred to the VM process")
- service_descriptor = b'test.Socket+ dom0\0'
- self.assertEqual(service_stdout, service_descriptor + TEST_DATA,
- 'Received data differs from what was sent')
- self.assertFalse(stderr,
- 'Some data was printed to stderr')
- self.assertFalse(service_stderr,
- 'Some data was printed to stderr')
- def test_096_qrexec_service_socket_vm_send(self):
- """Basic test socket services (VM) - send"""
- self.loop.run_until_complete(self.testvm1.start())
- self.create_remote_file(self.testvm1,
- '/tmp/service-input',
- TEST_DATA.decode())
- self.service_proc = self.loop.run_until_complete(self.testvm1.run(
- 'socat -u OPEN:/tmp/service-input UNIX-LISTEN:/etc/qubes-rpc/test.Socket,mode=666',
- user='root'))
- self._wait_for_socket_setup()
- try:
- (stdout, stderr) = self.loop.run_until_complete(asyncio.wait_for(
- self.testvm1.run_service_for_stdio('test.Socket+'),
- timeout=10))
- except subprocess.CalledProcessError as e:
- self.fail('{} exited with non-zero code {}; stderr: {}'.format(
- e.cmd, e.returncode, e.stderr))
- except asyncio.TimeoutError:
- self.fail(
- "service timeout, probably EOF wasn't transferred to the VM process")
- try:
- (service_stdout, service_stderr) = self.loop.run_until_complete(
- asyncio.wait_for(
- self.service_proc.communicate(),
- timeout=10))
- except asyncio.TimeoutError:
- self.fail(
- "socat timeout, probably EOF wasn't transferred to the VM process")
- self.assertEqual(stdout, TEST_DATA,
- 'Received data differs from what was sent')
- self.assertFalse(stderr,
- 'Some data was printed to stderr')
- self.assertFalse(service_stderr,
- 'Some data was printed to stderr')
- def test_097_qrexec_service_socket_vm_eof_reverse(self):
- """Test for EOF transmission VM(socket)->dom0"""
- self.loop.run_until_complete(self.testvm1.start())
- self.create_remote_file(self.testvm1,
- '/tmp/service_script',
- '#!/usr/bin/python3\n'
- 'import socket, os, sys, time\n'
- 's = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)\n'
- 'os.umask(0)\n'
- 's.bind("/etc/qubes-rpc/test.Socket")\n'
- 's.listen(1)\n'
- 'conn, addr = s.accept()\n'
- 'conn.send(b"test\\n")\n'
- 'conn.shutdown(socket.SHUT_WR)\n'
- # wait longer than the timeout below
- 'time.sleep(15)\n'
- )
- self.service_proc = self.loop.run_until_complete(self.testvm1.run(
- 'python3 /tmp/service_script',
- stdout=subprocess.PIPE, stdin=subprocess.PIPE,
- user='root'))
- self._wait_for_socket_setup()
- try:
- p = self.loop.run_until_complete(
- self.testvm1.run_service('test.Socket+',
- stdin=subprocess.PIPE, stdout=subprocess.PIPE))
- stdout = self.loop.run_until_complete(asyncio.wait_for(p.stdout.read(),
- timeout=10))
- except asyncio.TimeoutError:
- p.terminate()
- self.fail(
- "service timeout, probably EOF wasn't transferred from the VM process")
- finally:
- self.loop.run_until_complete(p.wait())
- self.assertEqual(stdout,
- b'test\n',
- 'Received data differs from what was expected')
- def test_098_qrexec_service_socket_vm_eof(self):
- """Test for EOF transmission dom0->VM(socket)"""
- self.loop.run_until_complete(self.testvm1.start())
- self.create_remote_file(
- self.testvm1,
- '/tmp/service_script',
- '#!/usr/bin/python3\n'
- 'import socket, os, sys, time\n'
- 's = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)\n'
- 'os.umask(0)\n'
- 's.bind("/etc/qubes-rpc/test.Socket")\n'
- 's.listen(1)\n'
- 'conn, addr = s.accept()\n'
- 'buf = conn.recv(100)\n'
- 'sys.stdout.buffer.write(buf)\n'
- 'buf = conn.recv(10)\n'
- 'sys.stdout.buffer.write(buf)\n'
- 'sys.stdout.buffer.flush()\n'
- 'os.close(1)\n'
- # wait longer than the timeout below
- 'time.sleep(15)\n'
- )
- self.service_proc = self.loop.run_until_complete(self.testvm1.run(
- 'python3 /tmp/service_script',
- stdout=subprocess.PIPE, stdin=subprocess.PIPE,
- user='root'))
- self._wait_for_socket_setup()
- try:
- p = self.loop.run_until_complete(
- self.testvm1.run_service('test.Socket+',
- stdin=subprocess.PIPE, stdout=subprocess.PIPE))
- p.stdin.write(b'test1test2')
- self.loop.run_until_complete(
- asyncio.wait_for(p.stdin.drain(), timeout=10))
- p.stdin.close()
- service_stdout = self.loop.run_until_complete(asyncio.wait_for(
- self.service_proc.stdout.read(),
- timeout=10))
- except asyncio.TimeoutError:
- p.terminate()
- self.fail(
- "service timeout, probably EOF wasn't transferred to the VM process")
- finally:
- self.loop.run_until_complete(p.wait())
- service_descriptor = b'test.Socket+ dom0\0'
- self.assertEqual(service_stdout, service_descriptor + b'test1test2',
- 'Received data differs from what was expected')
- def create_testcases_for_templates():
- return qubes.tests.create_testcases_for_templates('TC_00_Qrexec',
- TC_00_QrexecMixin, qubes.tests.SystemTestCase,
- module=sys.modules[__name__])
- def load_tests(loader, tests, pattern):
- tests.addTests(loader.loadTestsFromNames(
- create_testcases_for_templates()))
- return tests
- qubes.tests.maybe_create_testcases_on_import(create_testcases_for_templates)
|