test_message_spec.py
432 lines
| 11.3 KiB
| text/x-python
|
PythonLexer
MinRK
|
r16566 | """Test suite for our zeromq-based message specification.""" | ||
# Copyright (c) IPython Development Team. | ||||
# Distributed under the terms of the Modified BSD License. | ||||
Fernando Perez
|
r2926 | |||
MinRK
|
r6554 | import re | ||
Min RK
|
r19046 | import sys | ||
MinRK
|
r16566 | from distutils.version import LooseVersion as V | ||
Thomas Kluyver
|
r13354 | try: | ||
from queue import Empty # Py 3 | ||||
except ImportError: | ||||
from Queue import Empty # Py 2 | ||||
Fernando Perez
|
r2926 | |||
import nose.tools as nt | ||||
MinRK
|
r6554 | from IPython.utils.traitlets import ( | ||
Min RK
|
r19046 | HasTraits, TraitError, Bool, Unicode, Dict, Integer, List, Enum, | ||
MinRK
|
r6554 | ) | ||
Thomas Kluyver
|
r13361 | from IPython.utils.py3compat import string_types, iteritems | ||
MinRK
|
r6554 | |||
MinRK
|
r12414 | from .utils import TIMEOUT, start_global_kernel, flush_channels, execute | ||
MinRK
|
r6554 | #----------------------------------------------------------------------------- | ||
MinRK
|
r12414 | # Globals | ||
MinRK
|
r6554 | #----------------------------------------------------------------------------- | ||
MinRK
|
r12414 | KC = None | ||
MinRK
|
r11802 | |||
Fernando Perez
|
r2926 | def setup(): | ||
MinRK
|
r12414 | global KC | ||
KC = start_global_kernel() | ||||
MinRK
|
r6554 | |||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r12414 | # Message Spec References | ||
MinRK
|
r6554 | #----------------------------------------------------------------------------- | ||
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(): | ||||
Thomas Kluyver
|
r12374 | nt.assert_in(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
|
r16566 | assert False, str(e) | ||
MinRK
|
r6554 | |||
MinRK
|
r16665 | |||
MinRK
|
r16566 | class Version(Unicode): | ||
MinRK
|
r16665 | def __init__(self, *args, **kwargs): | ||
self.min = kwargs.pop('min', None) | ||||
self.max = kwargs.pop('max', None) | ||||
kwargs['default_value'] = self.min | ||||
super(Version, self).__init__(*args, **kwargs) | ||||
MinRK
|
r16566 | def validate(self, obj, value): | ||
MinRK
|
r16665 | if self.min and V(value) < V(self.min): | ||
raise TraitError("bad version: %s < %s" % (value, self.min)) | ||||
if self.max and (V(value) > V(self.max)): | ||||
raise TraitError("bad version: %s > %s" % (value, self.max)) | ||||
MinRK
|
r6554 | |||
class RMessage(Reference): | ||||
msg_id = Unicode() | ||||
msg_type = Unicode() | ||||
header = Dict() | ||||
parent_header = Dict() | ||||
content = Dict() | ||||
MinRK
|
r16566 | |||
def check(self, d): | ||||
super(RMessage, self).check(d) | ||||
RHeader().check(self.header) | ||||
MinRK
|
r16570 | if self.parent_header: | ||
RHeader().check(self.parent_header) | ||||
MinRK
|
r6554 | |||
class RHeader(Reference): | ||||
msg_id = Unicode() | ||||
msg_type = Unicode() | ||||
session = Unicode() | ||||
username = Unicode() | ||||
MinRK
|
r16665 | version = Version(min='5.0') | ||
MinRK
|
r6554 | |||
MinRK
|
r16665 | mime_pat = re.compile(r'^[\w\-\+\.]+/[\w\-\+\.]+$') | ||
MinRK
|
r16580 | |||
class MimeBundle(Reference): | ||||
metadata = Dict() | ||||
data = Dict() | ||||
def _data_changed(self, name, old, new): | ||||
for k,v in iteritems(new): | ||||
assert mime_pat.match(k) | ||||
nt.assert_is_instance(v, string_types) | ||||
# shell replies | ||||
MinRK
|
r6554 | |||
class ExecuteReply(Reference): | ||||
execution_count = Integer() | ||||
status = Enum((u'ok', u'error')) | ||||
def check(self, d): | ||||
Thomas Kluyver
|
r12374 | Reference.check(self, d) | ||
MinRK
|
r6554 | if d['status'] == 'ok': | ||
Thomas Kluyver
|
r12374 | ExecuteReplyOkay().check(d) | ||
MinRK
|
r6554 | elif d['status'] == 'error': | ||
Thomas Kluyver
|
r12374 | ExecuteReplyError().check(d) | ||
MinRK
|
r6554 | |||
Fernando Perez
|
r2926 | |||
MinRK
|
r6554 | class ExecuteReplyOkay(Reference): | ||
payload = List(Dict) | ||||
user_expressions = Dict() | ||||
Fernando Perez
|
r2926 | |||
MinRK
|
r6554 | |||
class ExecuteReplyError(Reference): | ||||
ename = Unicode() | ||||
evalue = Unicode() | ||||
traceback = List(Unicode) | ||||
MinRK
|
r16587 | class InspectReply(MimeBundle): | ||
MinRK
|
r6554 | found = Bool() | ||
class ArgSpec(Reference): | ||||
args = List(Unicode) | ||||
varargs = Unicode() | ||||
varkw = Unicode() | ||||
MinRK
|
r6558 | defaults = List() | ||
MinRK
|
r6554 | |||
class Status(Reference): | ||||
MinRK
|
r10341 | execution_state = Enum((u'busy', u'idle', u'starting')) | ||
MinRK
|
r6554 | |||
class CompleteReply(Reference): | ||||
matches = List(Unicode) | ||||
MinRK
|
r16588 | cursor_start = Integer() | ||
cursor_end = Integer() | ||||
status = Unicode() | ||||
MinRK
|
r6554 | |||
Min RK
|
r19046 | class LanguageInfo(Reference): | ||
name = Unicode('python') | ||||
version = Unicode(sys.version.split()[0]) | ||||
MinRK
|
r6554 | |||
Takafumi Arakaki
|
r8879 | class KernelInfoReply(Reference): | ||
MinRK
|
r16665 | protocol_version = Version(min='5.0') | ||
MinRK
|
r16584 | implementation = Unicode('ipython') | ||
MinRK
|
r16665 | implementation_version = Version(min='2.1') | ||
Thomas Kluyver
|
r18474 | language_info = Dict() | ||
MinRK
|
r16584 | banner = Unicode() | ||
Min RK
|
r19046 | |||
def check(self, d): | ||||
Reference.check(self, d) | ||||
LanguageInfo().check(d['language_info']) | ||||
Takafumi Arakaki
|
r8865 | |||
Takafumi Arakaki
|
r8831 | |||
Thomas Kluyver
|
r17622 | class IsCompleteReply(Reference): | ||
Thomas Kluyver
|
r17804 | status = Enum((u'complete', u'incomplete', u'invalid', u'unknown')) | ||
def check(self, d): | ||||
Reference.check(self, d) | ||||
if d['status'] == 'incomplete': | ||||
IsCompleteReplyIncomplete().check(d) | ||||
class IsCompleteReplyIncomplete(Reference): | ||||
indent = Unicode() | ||||
Thomas Kluyver
|
r17622 | |||
MinRK
|
r6554 | # IOPub messages | ||
MinRK
|
r16567 | class ExecuteInput(Reference): | ||
MinRK
|
r6554 | code = Unicode() | ||
MinRK
|
r6559 | execution_count = Integer() | ||
MinRK
|
r6554 | |||
MinRK
|
r16569 | Error = ExecuteReplyError | ||
MinRK
|
r6554 | |||
class Stream(Reference): | ||||
name = Enum((u'stdout', u'stderr')) | ||||
MinRK
|
r18104 | text = Unicode() | ||
MinRK
|
r6554 | |||
MinRK
|
r16580 | class DisplayData(MimeBundle): | ||
MinRK
|
r16585 | pass | ||
MinRK
|
r6554 | |||
MinRK
|
r16580 | class ExecuteResult(MimeBundle): | ||
MinRK
|
r6664 | execution_count = Integer() | ||
MinRK
|
r6554 | references = { | ||
'execute_reply' : ExecuteReply(), | ||||
MinRK
|
r16587 | 'inspect_reply' : InspectReply(), | ||
MinRK
|
r6554 | 'status' : Status(), | ||
'complete_reply' : CompleteReply(), | ||||
Takafumi Arakaki
|
r8879 | 'kernel_info_reply': KernelInfoReply(), | ||
Thomas Kluyver
|
r17622 | 'is_complete_reply': IsCompleteReply(), | ||
MinRK
|
r16567 | 'execute_input' : ExecuteInput(), | ||
MinRK
|
r16568 | 'execute_result' : ExecuteResult(), | ||
MinRK
|
r16569 | 'error' : Error(), | ||
MinRK
|
r6554 | 'stream' : Stream(), | ||
'display_data' : DisplayData(), | ||||
MinRK
|
r16566 | 'header' : RHeader(), | ||
MinRK
|
r6554 | } | ||
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: | ||||
Thomas Kluyver
|
r12374 | nt.assert_equal(msg['msg_type'], msg_type) | ||
MinRK
|
r6554 | if parent: | ||
Thomas Kluyver
|
r12374 | nt.assert_equal(msg['parent_header']['msg_id'], parent) | ||
MinRK
|
r6554 | content = msg['content'] | ||
ref = references[msg['msg_type']] | ||||
Thomas Kluyver
|
r12374 | ref.check(content) | ||
MinRK
|
r6554 | |||
#----------------------------------------------------------------------------- | ||||
# Tests | ||||
#----------------------------------------------------------------------------- | ||||
# Shell channel | ||||
Fernando Perez
|
r2926 | |||
def test_execute(): | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r10294 | msg_id = KC.execute(code='x=1') | ||
MinRK
|
r11802 | reply = KC.get_shell_msg(timeout=TIMEOUT) | ||
Thomas Kluyver
|
r12374 | validate_message(reply, 'execute_reply', msg_id) | ||
MinRK
|
r6554 | |||
def test_execute_silent(): | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | msg_id, reply = execute(code='x=1', silent=True) | ||
# flush status=idle | ||||
MinRK
|
r11802 | status = KC.iopub_channel.get_msg(timeout=TIMEOUT) | ||
Thomas Kluyver
|
r12374 | validate_message(status, 'status', msg_id) | ||
Bradley M. Froehle
|
r7875 | nt.assert_equal(status['content']['execution_state'], 'idle') | ||
MinRK
|
r6554 | |||
Thomas Kluyver
|
r12374 | nt.assert_raises(Empty, KC.iopub_channel.get_msg, timeout=0.1) | ||
MinRK
|
r6554 | count = reply['execution_count'] | ||
msg_id, reply = execute(code='x=2', silent=True) | ||||
# flush status=idle | ||||
MinRK
|
r11802 | status = KC.iopub_channel.get_msg(timeout=TIMEOUT) | ||
Thomas Kluyver
|
r12374 | validate_message(status, 'status', msg_id) | ||
nt.assert_equal(status['content']['execution_state'], 'idle') | ||||
MinRK
|
r6554 | |||
Thomas Kluyver
|
r12374 | nt.assert_raises(Empty, KC.iopub_channel.get_msg, timeout=0.1) | ||
MinRK
|
r6554 | count_2 = reply['execution_count'] | ||
Thomas Kluyver
|
r12374 | nt.assert_equal(count_2, count) | ||
MinRK
|
r6554 | |||
def test_execute_error(): | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | |||
msg_id, reply = execute(code='1/0') | ||||
Thomas Kluyver
|
r12374 | nt.assert_equal(reply['status'], 'error') | ||
nt.assert_equal(reply['ename'], 'ZeroDivisionError') | ||||
MinRK
|
r6554 | |||
MinRK
|
r16569 | error = KC.iopub_channel.get_msg(timeout=TIMEOUT) | ||
validate_message(error, 'error', msg_id) | ||||
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_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'] | ||||
MinRK
|
r10637 | nt.assert_equal(user_expressions, {u'foo': { | ||
u'status': u'ok', | ||||
u'data': {u'text/plain': u'2'}, | ||||
u'metadata': {}, | ||||
}}) | ||||
def test_user_expressions_fail(): | ||||
flush_channels() | ||||
msg_id, reply = execute(code='x=0', user_expressions=dict(foo='nosuchname')) | ||||
user_expressions = reply['user_expressions'] | ||||
foo = user_expressions['foo'] | ||||
nt.assert_equal(foo['status'], 'error') | ||||
nt.assert_equal(foo['ename'], 'NameError') | ||||
MinRK
|
r6554 | |||
def test_oinfo(): | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r16587 | msg_id = KC.inspect('a') | ||
MinRK
|
r11802 | reply = KC.get_shell_msg(timeout=TIMEOUT) | ||
MinRK
|
r16587 | validate_message(reply, 'inspect_reply', msg_id) | ||
MinRK
|
r6554 | |||
def test_oinfo_found(): | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | |||
msg_id, reply = execute(code='a=5') | ||||
MinRK
|
r16587 | msg_id = KC.inspect('a') | ||
MinRK
|
r11802 | reply = KC.get_shell_msg(timeout=TIMEOUT) | ||
MinRK
|
r16587 | validate_message(reply, 'inspect_reply', msg_id) | ||
MinRK
|
r6554 | content = reply['content'] | ||
Thomas Kluyver
|
r12374 | assert content['found'] | ||
MinRK
|
r16580 | text = content['data']['text/plain'] | ||
nt.assert_in('Type:', text) | ||||
nt.assert_in('Docstring:', text) | ||||
MinRK
|
r6554 | |||
MinRK
|
r6557 | def test_oinfo_detail(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6557 | |||
msg_id, reply = execute(code='ip=get_ipython()') | ||||
MinRK
|
r6558 | |||
MinRK
|
r16587 | msg_id = KC.inspect('ip.object_inspect', cursor_pos=10, detail_level=1) | ||
MinRK
|
r11802 | reply = KC.get_shell_msg(timeout=TIMEOUT) | ||
MinRK
|
r16587 | validate_message(reply, 'inspect_reply', msg_id) | ||
MinRK
|
r6557 | content = reply['content'] | ||
Thomas Kluyver
|
r12374 | assert content['found'] | ||
MinRK
|
r16580 | text = content['data']['text/plain'] | ||
nt.assert_in('Definition:', text) | ||||
nt.assert_in('Source:', text) | ||||
MinRK
|
r6557 | |||
def test_oinfo_not_found(): | ||||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6557 | |||
MinRK
|
r16587 | msg_id = KC.inspect('dne') | ||
MinRK
|
r11802 | reply = KC.get_shell_msg(timeout=TIMEOUT) | ||
MinRK
|
r16587 | validate_message(reply, 'inspect_reply', msg_id) | ||
MinRK
|
r6557 | content = reply['content'] | ||
Thomas Kluyver
|
r12374 | nt.assert_false(content['found']) | ||
MinRK
|
r6557 | |||
MinRK
|
r6554 | def test_complete(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | msg_id, reply = execute(code="alpha = albert = 5") | ||
MinRK
|
r16580 | msg_id = KC.complete('al', 2) | ||
MinRK
|
r11802 | reply = KC.get_shell_msg(timeout=TIMEOUT) | ||
Thomas Kluyver
|
r12374 | validate_message(reply, 'complete_reply', msg_id) | ||
MinRK
|
r6554 | matches = reply['content']['matches'] | ||
for name in ('alpha', 'albert'): | ||||
Thomas Kluyver
|
r12374 | nt.assert_in(name, matches) | ||
MinRK
|
r6554 | |||
Takafumi Arakaki
|
r8879 | def test_kernel_info_request(): | ||
Takafumi Arakaki
|
r8831 | flush_channels() | ||
MinRK
|
r10294 | msg_id = KC.kernel_info() | ||
MinRK
|
r11802 | reply = KC.get_shell_msg(timeout=TIMEOUT) | ||
Thomas Kluyver
|
r12374 | validate_message(reply, 'kernel_info_reply', msg_id) | ||
Takafumi Arakaki
|
r8831 | |||
Pablo de Oliveira
|
r12953 | def test_single_payload(): | ||
flush_channels() | ||||
msg_id, reply = execute(code="for i in range(3):\n"+ | ||||
" x=range?\n") | ||||
payload = reply['payload'] | ||||
next_input_pls = [pl for pl in payload if pl["source"] == "set_next_input"] | ||||
nt.assert_equal(len(next_input_pls), 1) | ||||
Thomas Kluyver
|
r17622 | def test_is_complete(): | ||
flush_channels() | ||||
msg_id = KC.is_complete("a = 1") | ||||
reply = KC.get_shell_msg(timeout=TIMEOUT) | ||||
validate_message(reply, 'is_complete_reply', msg_id) | ||||
Pablo de Oliveira
|
r12953 | |||
MinRK
|
r6561 | # IOPub channel | ||
MinRK
|
r6554 | def test_stream(): | ||
MinRK
|
r6558 | flush_channels() | ||
MinRK
|
r6554 | msg_id, reply = execute("print('hi')") | ||
MinRK
|
r11802 | stdout = KC.iopub_channel.get_msg(timeout=TIMEOUT) | ||
Thomas Kluyver
|
r12374 | validate_message(stdout, 'stream', msg_id) | ||
MinRK
|
r6554 | content = stdout['content'] | ||
MinRK
|
r18104 | nt.assert_equal(content['text'], u'hi\n') | ||
MinRK
|
r6554 | |||
MinRK
|
r6558 | def test_display_data(): | ||
flush_channels() | ||||
MinRK
|
r6554 | msg_id, reply = execute("from IPython.core.display import display; display(1)") | ||
MinRK
|
r11802 | display = KC.iopub_channel.get_msg(timeout=TIMEOUT) | ||
Thomas Kluyver
|
r12374 | validate_message(display, 'display_data', parent=msg_id) | ||
MinRK
|
r6554 | data = display['content']['data'] | ||
Thomas Kluyver
|
r12374 | nt.assert_equal(data['text/plain'], u'1') | ||
MinRK
|
r6554 | |||