Merge pull request #547 from tahoe-lafs/2969.aggressive-integration-test-cleanup

Aggressive integration test cleanup

Fixes: ticket:2969
This commit is contained in:
Jean-Paul Calderone 2019-02-18 09:45:33 -05:00 committed by GitHub
commit 2bf5228586
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 48 additions and 36 deletions

View File

@ -2,14 +2,13 @@ from __future__ import print_function
import sys import sys
import shutil import shutil
from sys import stdout as _stdout from time import sleep
from os import mkdir, listdir, unlink from os import mkdir, listdir
from os.path import join, abspath, curdir, exists from os.path import join, exists
from tempfile import mkdtemp, mktemp from tempfile import mkdtemp, mktemp
from functools import partial
from twisted.python.procutils import which from twisted.python.procutils import which
from twisted.internet.defer import Deferred, DeferredList
from twisted.internet.task import deferLater
from twisted.internet.error import ( from twisted.internet.error import (
ProcessExitedAlready, ProcessExitedAlready,
ProcessTerminated, ProcessTerminated,
@ -18,12 +17,15 @@ from twisted.internet.error import (
import pytest import pytest
import pytest_twisted import pytest_twisted
from util import _CollectOutputProtocol from util import (
from util import _MagicTextProtocol _CollectOutputProtocol,
from util import _DumpOutputProtocol _MagicTextProtocol,
from util import _ProcessExitedProtocol _DumpOutputProtocol,
from util import _create_node _ProcessExitedProtocol,
from util import _run_node _create_node,
_run_node,
_cleanup_twistd_process,
)
# pytest customization hooks # pytest customization hooks
@ -82,7 +84,7 @@ def flog_binary():
def flog_gatherer(reactor, temp_dir, flog_binary, request): def flog_gatherer(reactor, temp_dir, flog_binary, request):
out_protocol = _CollectOutputProtocol() out_protocol = _CollectOutputProtocol()
gather_dir = join(temp_dir, 'flog_gather') gather_dir = join(temp_dir, 'flog_gather')
process = reactor.spawnProcess( reactor.spawnProcess(
out_protocol, out_protocol,
flog_binary, flog_binary,
( (
@ -107,11 +109,7 @@ def flog_gatherer(reactor, temp_dir, flog_binary, request):
pytest_twisted.blockon(twistd_protocol.magic_seen) pytest_twisted.blockon(twistd_protocol.magic_seen)
def cleanup(): def cleanup():
try: _cleanup_twistd_process(twistd_process, twistd_protocol.exited)
twistd_process.signalProcess('TERM')
pytest_twisted.blockon(twistd_protocol.exited)
except ProcessExitedAlready:
pass
flog_file = mktemp('.flog_dump') flog_file = mktemp('.flog_dump')
flog_protocol = _DumpOutputProtocol(open(flog_file, 'w')) flog_protocol = _DumpOutputProtocol(open(flog_file, 'w'))
@ -126,7 +124,12 @@ def flog_gatherer(reactor, temp_dir, flog_binary, request):
'flogtool', 'dump', join(temp_dir, 'flog_gather', flogs[0]) 'flogtool', 'dump', join(temp_dir, 'flog_gather', flogs[0])
), ),
) )
pytest_twisted.blockon(flog_protocol.done) print("Waiting for flogtool to complete")
try:
pytest_twisted.blockon(flog_protocol.done)
except ProcessTerminated as e:
print("flogtool exited unexpectedly: {}".format(str(e)))
print("Flogtool completed")
request.addfinalizer(cleanup) request.addfinalizer(cleanup)
@ -180,14 +183,7 @@ log_gatherer.furl = {log_furl}
intro_dir, intro_dir,
), ),
) )
request.addfinalizer(partial(_cleanup_twistd_process, process, protocol.exited))
def cleanup():
try:
process.signalProcess('TERM')
pytest_twisted.blockon(protocol.exited)
except ProcessExitedAlready:
pass
request.addfinalizer(cleanup)
pytest_twisted.blockon(protocol.magic_seen) pytest_twisted.blockon(protocol.magic_seen)
return process return process
@ -198,7 +194,7 @@ def introducer_furl(introducer, temp_dir):
furl_fname = join(temp_dir, 'introducer', 'private', 'introducer.furl') furl_fname = join(temp_dir, 'introducer', 'private', 'introducer.furl')
while not exists(furl_fname): while not exists(furl_fname):
print("Don't see {} yet".format(furl_fname)) print("Don't see {} yet".format(furl_fname))
time.sleep(.1) sleep(.1)
furl = open(furl_fname, 'r').read() furl = open(furl_fname, 'r').read()
return furl return furl
@ -266,7 +262,7 @@ def tor_introducer_furl(tor_introducer, temp_dir):
furl_fname = join(temp_dir, 'introducer_tor', 'private', 'introducer.furl') furl_fname = join(temp_dir, 'introducer_tor', 'private', 'introducer.furl')
while not exists(furl_fname): while not exists(furl_fname):
print("Don't see {} yet".format(furl_fname)) print("Don't see {} yet".format(furl_fname))
time.sleep(.1) sleep(.1)
furl = open(furl_fname, 'r').read() furl = open(furl_fname, 'r').read()
return furl return furl
@ -377,7 +373,7 @@ def magic_folder(reactor, alice_invite, alice, bob, temp_dir, request):
print("pairing magic-folder") print("pairing magic-folder")
bob_dir = join(temp_dir, 'bob') bob_dir = join(temp_dir, 'bob')
proto = _CollectOutputProtocol() proto = _CollectOutputProtocol()
transport = reactor.spawnProcess( reactor.spawnProcess(
proto, proto,
sys.executable, sys.executable,
[ [

View File

@ -3,6 +3,7 @@ import time
from os import mkdir from os import mkdir
from os.path import exists, join from os.path import exists, join
from StringIO import StringIO from StringIO import StringIO
from functools import partial
from twisted.internet.defer import Deferred, succeed from twisted.internet.defer import Deferred, succeed
from twisted.internet.protocol import ProcessProtocol from twisted.internet.protocol import ProcessProtocol
@ -105,6 +106,26 @@ class _MagicTextProtocol(ProcessProtocol):
sys.stdout.write(data) sys.stdout.write(data)
def _cleanup_twistd_process(twistd_process, exited):
"""
Terminate the given process with a kill signal (SIGKILL on POSIX,
TerminateProcess on Windows).
:param twistd_process: The `IProcessTransport` representing the process.
:param exited: A `Deferred` which fires when the process has exited.
:return: After the process has exited.
"""
try:
print("signaling {} with KILL".format(twistd_process.pid))
twistd_process.signalProcess('KILL')
print("signaled, blocking on exit")
pytest_twisted.blockon(exited)
print("exited, goodbye")
except ProcessExitedAlready:
pass
def _run_node(reactor, node_dir, request, magic_text): def _run_node(reactor, node_dir, request, magic_text):
if magic_text is None: if magic_text is None:
magic_text = "client running" magic_text = "client running"
@ -124,13 +145,7 @@ def _run_node(reactor, node_dir, request, magic_text):
) )
process.exited = protocol.exited process.exited = protocol.exited
def cleanup(): request.addfinalizer(partial(_cleanup_twistd_process, process, protocol.exited))
try:
process.signalProcess('TERM')
pytest_twisted.blockon(protocol.exited)
except ProcessExitedAlready:
pass
request.addfinalizer(cleanup)
# we return the 'process' ITransport instance # we return the 'process' ITransport instance
# XXX abusing the Deferred; should use .when_magic_seen() or something? # XXX abusing the Deferred; should use .when_magic_seen() or something?

1
newsfragments/2969.other Normal file
View File

@ -0,0 +1 @@
The magic-folder integration test suite now performs more aggressive cleanup of the processes it launches.