test_message_spec.py
497 lines
| 13.1 KiB
| text/x-python
|
PythonLexer
Fernando Perez
|
r2926 | """Test suite for our zeromq-based messaging specification. | ||
""" | ||||
#----------------------------------------------------------------------------- | ||||
Matthias BUSSONNIER
|
r5390 | # Copyright (C) 2010-2011 The IPython Development Team | ||
Fernando Perez
|
r2926 | # | ||
# Distributed under the terms of the BSD License. The full license is in | ||||
# the file COPYING.txt, distributed as part of this software. | ||||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r6554 | import re | ||
Fernando Perez
|
r2926 | import sys | ||
import time | ||||
MinRK
|
r6554 | from subprocess import PIPE | ||
from Queue import Empty | ||||
Fernando Perez
|
r2926 | |||
import nose.tools as nt | ||||
from ..blockingkernelmanager import BlockingKernelManager | ||||
MinRK
|
r6558 | |||
from IPython.testing import decorators as dec | ||||
Fernando Perez
|
r2926 | from IPython.utils import io | ||
MinRK
|
r6554 | from IPython.utils.traitlets import ( | ||
Takafumi Arakaki
|
r8865 | HasTraits, TraitError, Bool, Unicode, Dict, Integer, List, Enum, Any, | ||
MinRK
|
r6554 | ) | ||
#----------------------------------------------------------------------------- | ||||
# Global setup and utilities | ||||
#----------------------------------------------------------------------------- | ||||
Fernando Perez
|
r2926 | |||
def setup(): | ||||
global KM | ||||
KM = BlockingKernelManager() | ||||
MinRK
|
r6554 | KM.start_kernel(stdout=PIPE, stderr=PIPE) | ||
Fernando Perez
|
r2926 | KM.start_channels() | ||
MinRK
|
r6664 | |||
# wait for kernel to be ready | ||||
KM.shell_channel.execute("pass") | ||||
KM.shell_channel.get_msg(block=True, timeout=5) | ||||
flush_channels() | ||||
Fernando Perez
|
r2926 | |||
MinRK
|
r6558 | |||
Fernando Perez
|
r2926 | def teardown(): | ||
KM.stop_channels() | ||||
MinRK
|
r6554 | KM.shutdown_kernel() | ||
MinRK
|
r6558 | |||
MinRK
|
r6554 | def flush_channels(): | ||
"""flush any messages waiting on the queue""" | ||||
for channel in (KM.shell_channel, KM.sub_channel): | ||||
MinRK
|
r6558 | while True: | ||
try: | ||||
msg = channel.get_msg(block=True, timeout=0.1) | ||||
except Empty: | ||||
break | ||||
else: | ||||
MinRK
|
r6573 | list(validate_message(msg)) | ||
MinRK
|
r6554 | |||
def execute(code='', **kwargs): | ||||
"""wrapper for doing common steps for validating an execution request""" | ||||
shell = KM.shell_channel | ||||
sub = KM.sub_channel | ||||
msg_id = shell.execute(code=code, **kwargs) | ||||
reply = shell.get_msg(timeout=2) | ||||
MinRK
|
r6573 | list(validate_message(reply, 'execute_reply', msg_id)) | ||
MinRK
|
r6554 | busy = sub.get_msg(timeout=2) | ||
MinRK
|
r6573 | list(validate_message(busy, 'status', msg_id)) | ||
Bradley M. Froehle
|
r7875 | nt.assert_equal(busy['content']['execution_state'], 'busy') | ||
MinRK
|
r6554 | |||
if not kwargs.get('silent'): | ||||
pyin = sub.get_msg(timeout=2) | ||||
MinRK
|
r6573 | list(validate_message(pyin, 'pyin', msg_id)) | ||
Bradley M. Froehle
|
r7875 | nt.assert_equal(pyin['content']['code'], code) | ||
MinRK
|
r6554 | |||
return msg_id, reply['content'] | ||||
#----------------------------------------------------------------------------- | ||||
# MSG Spec References | ||||
#----------------------------------------------------------------------------- | ||||
class Reference(HasTraits): | ||||
Takafumi Arakaki
|
r8829 | |||
""" | ||||
Base class for message spec specification testing. | ||||
This class is the core of the message specification test. The | ||||
idea is that child classes implement trait attributes for each | ||||
message keys, so that message keys can be tested against these | ||||
traits using :meth:`check` method. | ||||
""" | ||||
MinRK
|
r6554 | def check(self, d): | ||
"""validate a dict against our traits""" | ||||
for key in self.trait_names(): | ||||
MinRK
|
r6558 | yield nt.assert_true(key in d, "Missing key: %r, should be found in %s" % (key, d)) | ||
MinRK
|
r6554 | # FIXME: always allow None, probably not a good idea | ||
if d[key] is None: | ||||
continue | ||||
try: | ||||
setattr(self, key, d[key]) | ||||
except TraitError as e: | ||||
MinRK
|
r6558 | yield nt.assert_true(False, str(e)) | ||
MinRK
|
r6554 | |||
class RMessage(Reference): | ||||
msg_id = Unicode() | ||||
msg_type = Unicode() | ||||
header = Dict() | ||||
parent_header = Dict() | ||||
content = Dict() | ||||
class RHeader(Reference): | ||||
msg_id = Unicode() | ||||
msg_type = Unicode() | ||||
session = Unicode() | ||||
username = Unicode() | ||||
class RContent(Reference): | ||||
status = Enum((u'ok', u'error')) | ||||
class ExecuteReply(Reference): | ||||
execution_count = Integer() | ||||
status = Enum((u'ok', u'error')) | ||||
def check(self, d): | ||||
MinRK
|
r6558 | for tst in Reference.check(self, d): | ||
yield tst | ||||
MinRK
|
r6554 | if d['status'] == 'ok': | ||
MinRK
|
r6558 | for tst in ExecuteReplyOkay().check(d): | ||
yield tst | ||||
MinRK
|
r6554 | elif d['status'] == 'error': | ||
MinRK
|
r6558 | for tst in ExecuteReplyError().check(d): | ||
yield tst | ||||
MinRK
|
r6554 | |||
Fernando Perez
|
r2926 | |||
MinRK
|
r6554 | class ExecuteReplyOkay(Reference): | ||
payload = List(Dict) | ||||
user_variables = Dict() | ||||
user_expressions = Dict() | ||||
Fernando Perez
|
r2926 | |||
MinRK
|
r6554 | |||
class ExecuteReplyError(Reference): | ||||
ename = Unicode() | ||||
evalue = Unicode() | ||||
traceback = List(Unicode) | ||||
class OInfoReply(Reference): | ||||
name = Unicode() | ||||
found = Bool() | ||||
ismagic = Bool() | ||||
isalias = Bool() | ||||
namespace = Enum((u'builtin', u'magics', u'alias', u'Interactive')) | ||||
type_name = Unicode() | ||||
string_form = Unicode() | ||||
base_class = Unicode() | ||||
length = Integer() | ||||
file = Unicode() | ||||
definition = Unicode() | ||||
argspec = Dict() | ||||
init_definition = Unicode() | ||||
docstring = Unicode() | ||||
init_docstring = Unicode() | ||||
class_docstring = Unicode() | ||||
call_def = Unicode() | ||||
call_docstring = Unicode() | ||||
source = Unicode() | ||||
def check(self, d): | ||||
MinRK
|
r6558 | for tst in Reference.check(self, d): | ||
yield tst | ||||
MinRK
|
r6554 | if d['argspec'] is not None: | ||
MinRK
|
r6558 | for tst in ArgSpec().check(d['argspec']): | ||
yield tst | ||||
MinRK
|
r6554 | |||
class ArgSpec(Reference): | ||||
args = List(Unicode) | ||||
varargs = Unicode() | ||||
varkw = Unicode() | ||||
MinRK
|
r6558 | defaults = List() | ||
MinRK
|
r6554 | |||
class Status(Reference): | ||||
execution_state = Enum((u'busy', u'idle')) | ||||
class CompleteReply(Reference): | ||||
matches = List(Unicode) | ||||
Takafumi Arakaki
|
r8865 | def Version(num, trait=Integer): | ||
return List(trait, default_value=[0] * num, minlen=num, maxlen=num) | ||||
Takafumi Arakaki
|
r8831 | class VersionReply(Reference): | ||
Takafumi Arakaki
|
r8865 | |||
protocol_version = Version(2) | ||||
ipython_version = Version(4, Any) | ||||
language_version = Version(3) | ||||
language = Unicode() | ||||
def _ipython_version_changed(self, name, old, new): | ||||
for v in new: | ||||
nt.assert_true( | ||||
isinstance(v, int) or isinstance(v, basestring), | ||||
Takafumi Arakaki
|
r8875 | 'expected int or string as version component, got {0!r}' | ||
Takafumi Arakaki
|
r8865 | .format(v)) | ||
Takafumi Arakaki
|
r8831 | |||
MinRK
|
r6554 | # IOPub messages | ||
class PyIn(Reference): | ||||
code = Unicode() | ||||
MinRK
|
r6559 | execution_count = Integer() | ||
MinRK
|
r6554 | |||
PyErr = ExecuteReplyError | ||||
class Stream(Reference): | ||||
name = Enum((u'stdout', u'stderr')) | ||||
data = Unicode() | ||||
mime_pat = re.compile(r'\w+/\w+') | ||||
class DisplayData(Reference): | ||||
source = Unicode() | ||||
metadata = Dict() | ||||
data = Dict() | ||||
def _data_changed(self, name, old, new): | ||||
for k,v in new.iteritems(): | ||||
nt.assert_true(mime_pat.match(k)) | ||||
nt.assert_true(isinstance(v, basestring), "expected string data, got %r" % v) | ||||
MinRK
|
r6664 | class PyOut(Reference): | ||
execution_count = Integer() | ||||
data = Dict() | ||||
def _data_changed(self, name, old, new): | ||||
for k,v in new.iteritems(): | ||||
nt.assert_true(mime_pat.match(k)) | ||||
nt.assert_true(isinstance(v, basestring), "expected string data, got %r" % v) | ||||
MinRK
|
r6554 | references = { | ||
'execute_reply' : ExecuteReply(), | ||||
'object_info_reply' : OInfoReply(), | ||||
'status' : Status(), | ||||
'complete_reply' : CompleteReply(), | ||||
Takafumi Arakaki
|
r8831 | 'version_reply': VersionReply(), | ||
MinRK
|
r6554 | 'pyin' : PyIn(), | ||
MinRK
|
r6664 | 'pyout' : PyOut(), | ||
MinRK
|
r6554 | 'pyerr' : PyErr(), | ||
'stream' : Stream(), | ||||
'display_data' : DisplayData(), | ||||
} | ||||
Takafumi Arakaki
|
r8829 | """ | ||
Specifications of `content` part of the reply messages. | ||||
""" | ||||
MinRK
|
r6554 | |||
def validate_message(msg, msg_type=None, parent=None): | ||||
MinRK
|
r6573 | """validate a message | ||
This is a generator, and must be iterated through to actually | ||||
trigger each test. | ||||
If msg_type and/or parent are given, the msg_type and/or parent msg_id | ||||
are compared with the given values. | ||||
""" | ||||
MinRK
|
r6554 | RMessage().check(msg) | ||
if msg_type: | ||||
Bradley M. Froehle
|
r7875 | yield nt.assert_equal(msg['msg_type'], msg_type) | ||
MinRK
|
r6554 | if parent: | ||
MinRK
|
r6558 | yield nt.assert_equal(msg['parent_header']['msg_id'], parent) | ||
MinRK
|
r6554 | content = msg['content'] | ||
ref = references[msg['msg_type']] | ||||
MinRK
|
r6558 | for tst in ref.check(content): | ||
yield tst | ||||
MinRK
|
r6554 | |||
#----------------------------------------------------------------------------- | ||||
# Tests | ||||
#----------------------------------------------------------------------------- | ||||
# Shell channel | ||||
Fernando Perez
|
r2926 | |||
MinRK
|
r6558 | @dec.parametric | ||
Fernando Perez
|
r2926 | def test_execute(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | shell = KM.shell_channel | ||
msg_id = shell.execute(code='x=1') | ||||
reply = shell.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(reply, 'execute_reply', msg_id): | ||
yield tst | ||||
MinRK
|
r6554 | |||
MinRK
|
r6558 | @dec.parametric | ||
MinRK
|
r6554 | def test_execute_silent(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | msg_id, reply = execute(code='x=1', silent=True) | ||
# flush status=idle | ||||
status = KM.sub_channel.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(status, 'status', msg_id): | ||
yield tst | ||||
Bradley M. Froehle
|
r7875 | nt.assert_equal(status['content']['execution_state'], 'idle') | ||
MinRK
|
r6554 | |||
MinRK
|
r6558 | yield nt.assert_raises(Empty, KM.sub_channel.get_msg, timeout=0.1) | ||
MinRK
|
r6554 | count = reply['execution_count'] | ||
msg_id, reply = execute(code='x=2', silent=True) | ||||
# flush status=idle | ||||
status = KM.sub_channel.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(status, 'status', msg_id): | ||
yield tst | ||||
Bradley M. Froehle
|
r7875 | yield nt.assert_equal(status['content']['execution_state'], 'idle') | ||
MinRK
|
r6554 | |||
MinRK
|
r6558 | yield nt.assert_raises(Empty, KM.sub_channel.get_msg, timeout=0.1) | ||
MinRK
|
r6554 | count_2 = reply['execution_count'] | ||
Bradley M. Froehle
|
r7875 | yield nt.assert_equal(count_2, count) | ||
MinRK
|
r6554 | |||
MinRK
|
r6558 | @dec.parametric | ||
MinRK
|
r6554 | def test_execute_error(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | |||
msg_id, reply = execute(code='1/0') | ||||
Bradley M. Froehle
|
r7875 | yield nt.assert_equal(reply['status'], 'error') | ||
yield nt.assert_equal(reply['ename'], 'ZeroDivisionError') | ||||
MinRK
|
r6554 | |||
pyerr = KM.sub_channel.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(pyerr, 'pyerr', msg_id): | ||
yield tst | ||||
MinRK
|
r6554 | |||
def test_execute_inc(): | ||||
"""execute request should increment execution_count""" | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | msg_id, reply = execute(code='x=1') | ||
count = reply['execution_count'] | ||||
flush_channels() | ||||
Fernando Perez
|
r2926 | |||
MinRK
|
r6554 | msg_id, reply = execute(code='x=2') | ||
count_2 = reply['execution_count'] | ||||
Bradley M. Froehle
|
r7875 | nt.assert_equal(count_2, count+1) | ||
MinRK
|
r6554 | |||
def test_user_variables(): | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | msg_id, reply = execute(code='x=1', user_variables=['x']) | ||
user_variables = reply['user_variables'] | ||||
Bradley M. Froehle
|
r7875 | nt.assert_equal(user_variables, {u'x' : u'1'}) | ||
MinRK
|
r6554 | |||
def test_user_expressions(): | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | msg_id, reply = execute(code='x=1', user_expressions=dict(foo='x+1')) | ||
user_expressions = reply['user_expressions'] | ||||
Bradley M. Froehle
|
r7875 | nt.assert_equal(user_expressions, {u'foo' : u'2'}) | ||
MinRK
|
r6554 | |||
MinRK
|
r6558 | @dec.parametric | ||
MinRK
|
r6554 | def test_oinfo(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | shell = KM.shell_channel | ||
msg_id = shell.object_info('a') | ||||
reply = shell.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(reply, 'object_info_reply', msg_id): | ||
yield tst | ||||
MinRK
|
r6554 | |||
MinRK
|
r6558 | @dec.parametric | ||
MinRK
|
r6554 | def test_oinfo_found(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | |||
MinRK
|
r6558 | shell = KM.shell_channel | ||
MinRK
|
r6554 | msg_id, reply = execute(code='a=5') | ||
msg_id = shell.object_info('a') | ||||
reply = shell.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(reply, 'object_info_reply', msg_id): | ||
yield tst | ||||
MinRK
|
r6554 | content = reply['content'] | ||
MinRK
|
r6558 | yield nt.assert_true(content['found']) | ||
MinRK
|
r6560 | argspec = content['argspec'] | ||
yield nt.assert_true(argspec is None, "didn't expect argspec dict, got %r" % argspec) | ||||
MinRK
|
r6554 | |||
MinRK
|
r6558 | @dec.parametric | ||
MinRK
|
r6557 | def test_oinfo_detail(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6557 | |||
MinRK
|
r6558 | shell = KM.shell_channel | ||
MinRK
|
r6557 | msg_id, reply = execute(code='ip=get_ipython()') | ||
MinRK
|
r6558 | |||
MinRK
|
r6557 | msg_id = shell.object_info('ip.object_inspect', detail_level=2) | ||
reply = shell.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(reply, 'object_info_reply', msg_id): | ||
yield tst | ||||
MinRK
|
r6557 | content = reply['content'] | ||
MinRK
|
r6558 | yield nt.assert_true(content['found']) | ||
MinRK
|
r6560 | argspec = content['argspec'] | ||
yield nt.assert_true(isinstance(argspec, dict), "expected non-empty argspec dict, got %r" % argspec) | ||||
Bradley M. Froehle
|
r7875 | yield nt.assert_equal(argspec['defaults'], [0]) | ||
MinRK
|
r6557 | |||
MinRK
|
r6558 | @dec.parametric | ||
MinRK
|
r6557 | def test_oinfo_not_found(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6557 | |||
MinRK
|
r6558 | shell = KM.shell_channel | ||
MinRK
|
r6557 | msg_id = shell.object_info('dne') | ||
reply = shell.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(reply, 'object_info_reply', msg_id): | ||
yield tst | ||||
MinRK
|
r6557 | content = reply['content'] | ||
MinRK
|
r6558 | yield nt.assert_false(content['found']) | ||
MinRK
|
r6557 | |||
MinRK
|
r6558 | @dec.parametric | ||
MinRK
|
r6554 | def test_complete(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | shell = KM.shell_channel | ||
msg_id, reply = execute(code="alpha = albert = 5") | ||||
msg_id = shell.complete('al', 'al', 2) | ||||
reply = shell.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(reply, 'complete_reply', msg_id): | ||
yield tst | ||||
MinRK
|
r6554 | matches = reply['content']['matches'] | ||
for name in ('alpha', 'albert'): | ||||
MinRK
|
r6558 | yield nt.assert_true(name in matches, "Missing match: %r" % name) | ||
MinRK
|
r6554 | |||
Takafumi Arakaki
|
r8831 | @dec.parametric | ||
def test_version_request(): | ||||
flush_channels() | ||||
shell = KM.shell_channel | ||||
msg_id = shell.version() | ||||
reply = shell.get_msg(timeout=2) | ||||
for tst in validate_message(reply, 'version_reply', msg_id): | ||||
yield tst | ||||
MinRK
|
r6561 | # IOPub channel | ||
MinRK
|
r6558 | @dec.parametric | ||
MinRK
|
r6554 | def test_stream(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | msg_id, reply = execute("print('hi')") | ||
stdout = KM.sub_channel.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(stdout, 'stream', msg_id): | ||
yield tst | ||||
MinRK
|
r6554 | content = stdout['content'] | ||
Bradley M. Froehle
|
r7875 | yield nt.assert_equal(content['name'], u'stdout') | ||
yield nt.assert_equal(content['data'], u'hi\n') | ||||
MinRK
|
r6554 | |||
MinRK
|
r6558 | @dec.parametric | ||
def test_display_data(): | ||||
flush_channels() | ||||
MinRK
|
r6554 | msg_id, reply = execute("from IPython.core.display import display; display(1)") | ||
display = KM.sub_channel.get_msg(timeout=2) | ||||
MinRK
|
r6558 | for tst in validate_message(display, 'display_data', parent=msg_id): | ||
yield tst | ||||
MinRK
|
r6554 | data = display['content']['data'] | ||
Bradley M. Froehle
|
r7875 | yield nt.assert_equal(data['text/plain'], u'1') | ||
MinRK
|
r6554 | |||