Skip to content

Commit

Permalink
Making emulator scripts wait until emulators are ready.
Browse files Browse the repository at this point in the history
Also manually handling clean-up of child processes spawned.
The ``gcloud`` tool was orphan-ing some processes.
  • Loading branch information
dhermes committed Jan 27, 2016
1 parent 2096618 commit 31e6643
Show file tree
Hide file tree
Showing 3 changed files with 86 additions and 13 deletions.
16 changes: 9 additions & 7 deletions CONTRIBUTING.rst
Original file line number Diff line number Diff line change
Expand Up @@ -222,7 +222,7 @@ Running System Tests
``tox`` environment), first start the emulator and
take note of the process ID::

$ gcloud beta emulators datastore start &
$ gcloud beta emulators datastore start &> log.txt &
[1] 33333

then determine the environment variables needed to interact with
Expand All @@ -242,9 +242,10 @@ Running System Tests
> python system_tests/run_system_test.py \
> --package=datastore --ignore-requirements

and after completion stop the emulator::
and after completion stop the emulator and any child
processes it spawned::

$ kill 33333
$ kill -- -33333

.. _emulators: https://cloud.google.com/sdk/gcloud/reference/beta/emulators/

Expand All @@ -255,24 +256,25 @@ Running System Tests
If you'd like to run them directly (outside of a ``tox`` environment), first
start the emulator and take note of the process ID::

$ gcloud beta emulators pubsub start &
$ gcloud beta emulators pubsub start &> log.txt &
[1] 44444

then determine the environment variables needed to interact with
the emulator::

$ gcloud beta emulators pubsub env-init
export PUBSUB_EMULATOR_HOST=localhost:8897
export PUBSUB_EMULATOR_HOST=localhost:8897

using these environment variables run the emulator::

$ PUBSUB_EMULATOR_HOST=http://localhost:8897 \
> python system_tests/run_system_test.py \
> --package=pubsub

and after completion stop the emulator::
and after completion stop the emulator and any child
processes it spawned::

$ kill 44444
$ kill -- -44444

Test Coverage
-------------
Expand Down
78 changes: 72 additions & 6 deletions scripts/run_emulator.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,8 @@
import os
import subprocess

import psutil

from gcloud.environment_vars import GCD_DATASET
from gcloud.environment_vars import GCD_HOST
from gcloud.environment_vars import PUBSUB_EMULATOR
Expand All @@ -33,6 +35,8 @@
'datastore': (GCD_DATASET, GCD_HOST),
'pubsub': (PUBSUB_EMULATOR,)
}
_DS_READY_LINE = '[datastore] INFO: Dev App Server is now running\n'
_PS_READY_LINE_PREFIX = '[pubsub] INFO: Server started, listening on '


def get_parser():
Expand Down Expand Up @@ -73,6 +77,66 @@ def get_env_init_command(package):
return ('gcloud', 'beta', 'emulators', package, 'env-init')


def datastore_wait_ready(popen):
"""Wait until the datastore emulator is ready to use.
:type popen: :class:`subprocess.Popen`
:param popen: An open subprocess to interact with.
"""
emulator_ready = False
while not emulator_ready:
emulator_ready = popen.stderr.readline() == _DS_READY_LINE


def pubsub_wait_ready(popen):
"""Wait until the pubsub emulator is ready to use.
:type popen: :class:`subprocess.Popen`
:param popen: An open subprocess to interact with.
"""
emulator_ready = False
while not emulator_ready:
emulator_ready = popen.stderr.readline().startswith(
_PS_READY_LINE_PREFIX)


def wait_ready(package, popen):
"""Wait until the emulator is ready to use.
:type package: str
:param package: The package to check if ready.
:type popen: :class:`subprocess.Popen`
:param popen: An open subprocess to interact with.
:raises: :class:`KeyError` if the ``package`` is not among
``datastore``, ``pubsub``.
"""
if package == 'datastore':
datastore_wait_ready(popen)
elif package == 'pubsub':
pubsub_wait_ready(popen)
else:
raise KeyError('')


def cleanup(pid):
"""Cleanup a process (including all of its children).
:type pid: int
:param pid: Process ID.
"""
proc = psutil.Process(pid)
for child_proc in proc.children(recursive=True):
try:
child_proc.kill()
child_proc.terminate()
except psutil.NoSuchProcess:
pass
proc.terminate()
proc.kill()


def run_tests_in_emulator(package):
"""Spawn an emulator instance and run the system tests.
Expand All @@ -87,9 +151,14 @@ def run_tests_in_emulator(package):
proc_start = subprocess.Popen(start_command, stdout=subprocess.PIPE,
stderr=subprocess.PIPE)
try:
wait_ready(package, proc_start)
env_init_command = get_env_init_command(package)
env_lines = subprocess.check_output(
env_init_command).strip().split('\n')
proc_env = subprocess.Popen(env_init_command, stdout=subprocess.PIPE,
stderr=subprocess.PIPE)
env_status = proc_env.wait()
if env_status != 0:
raise RuntimeError(env_status, proc_env.stderr.read())
env_lines = proc_env.stdout.read().strip().split('\n')
# Set environment variables before running the system tests.
for env_var in env_vars:
line_prefix = 'export ' + env_var + '='
Expand All @@ -99,10 +168,7 @@ def run_tests_in_emulator(package):
run_module_tests(package,
ignore_requirements=True)
finally:
# NOTE: This is mostly defensive. Since ``proc_start`` will be spawned
# by this current process, it should be killed when this process
# exits whether or not we kill it.
proc_start.kill()
cleanup(proc_start.pid)


def main():
Expand Down
5 changes: 5 additions & 0 deletions tox.ini
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ deps =
pep8
pylint
unittest2
psutil
passenv = {[testenv:system-tests]passenv}

[testenv:system-tests]
Expand All @@ -98,10 +99,14 @@ commands =
setenv =
PYTHONPATH = {toxinidir}/_testing
GCLOUD_NO_PRINT=true
deps =
{[testenv]deps}
psutil

[testenv:pubsub-emulator]
basepython =
python2.7
commands =
python {toxinidir}/scripts/run_emulator.py --package=pubsub
passenv = GCLOUD_*
deps = {[testenv:datastore-emulator]deps}

0 comments on commit 31e6643

Please sign in to comment.