2018-01-05 16:57:38 +00:00
|
|
|
import sys
|
2018-01-18 14:47:27 +00:00
|
|
|
|
2018-06-02 10:36:04 +01:00
|
|
|
from typing import List
|
|
|
|
from traitlets import Bool
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-03-12 15:11:59 +00:00
|
|
|
|
2018-12-21 20:57:53 +00:00
|
|
|
from yap4py.systuples import *
|
2018-03-12 15:11:59 +00:00
|
|
|
from yap4py.yapi import *
|
2018-10-19 16:19:53 +01:00
|
|
|
from IPython.core.completer import Completer
|
|
|
|
# import IPython.core
|
|
|
|
from IPython.core.inputsplitter import *
|
|
|
|
from IPython.core.inputtransformer import *
|
|
|
|
from IPython.core.interactiveshell import *
|
2018-10-16 14:33:16 +01:00
|
|
|
from ipython_genutils.py3compat import builtin_mod
|
2018-12-05 09:09:19 +00:00
|
|
|
|
2019-04-06 10:07:54 +01:00
|
|
|
import copy
|
2019-02-27 04:23:21 +00:00
|
|
|
import json
|
|
|
|
|
2018-12-05 09:09:19 +00:00
|
|
|
from yap_kernel.displayhook import ZMQShellDisplayHook
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-07-23 17:13:51 +01:00
|
|
|
import traceback
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-03-17 10:38:56 +00:00
|
|
|
global engine
|
2018-01-18 14:47:27 +00:00
|
|
|
|
2018-06-01 08:37:25 +01:00
|
|
|
def tracefunc(frame, event, arg, indent=[0]):
|
|
|
|
if event == "call":
|
|
|
|
indent[0] += 2
|
|
|
|
print( "-" * indent[0] + "> call function", frame.f_code.co_name )
|
|
|
|
elif event == "return":
|
|
|
|
print( "<" + "-" * indent[0], "exit function", frame.f_code.co_name )
|
|
|
|
indent[0] -= 2
|
|
|
|
return tracefunc
|
|
|
|
|
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
class YAPInputSplitter(InputSplitter):
|
|
|
|
"""An input splitter that recognizes all of iyap's special syntax."""
|
|
|
|
|
|
|
|
# String with raw, untransformed input.
|
|
|
|
source_raw = ''
|
|
|
|
|
|
|
|
# Flag to track when a transformer has stored input that it hasn't given
|
|
|
|
# back yet.
|
|
|
|
transformer_accumulating = False
|
|
|
|
|
|
|
|
# Flag to track when assemble_yap_lines has stored input that it hasn't
|
|
|
|
# given back yet.
|
|
|
|
within_yap_line = False
|
|
|
|
|
|
|
|
# Private attributes
|
|
|
|
|
|
|
|
# List with lines of raw input accumulated so far.
|
|
|
|
_buffer_raw = None
|
|
|
|
|
2018-03-14 00:41:05 +00:00
|
|
|
def __init__(self, engine=None, shell=None, line_input_checker=True, physical_line_transforms=None,
|
2018-01-18 14:47:27 +00:00
|
|
|
logical_line_transforms=None):
|
|
|
|
self._buffer_raw = []
|
2018-01-29 15:24:32 +00:00
|
|
|
self._validate = True
|
2018-12-21 20:57:53 +00:00
|
|
|
self.engine = engine
|
2018-03-14 00:41:05 +00:00
|
|
|
self.shell = shell
|
2018-01-18 14:47:27 +00:00
|
|
|
|
|
|
|
if physical_line_transforms is not None:
|
|
|
|
self.physical_line_transforms = physical_line_transforms
|
|
|
|
else:
|
|
|
|
self.physical_line_transforms = [
|
|
|
|
leading_indent(),
|
|
|
|
classic_prompt(),
|
|
|
|
ipy_prompt(),
|
|
|
|
cellmagic(end_on_blank_line=line_input_checker),
|
|
|
|
]
|
|
|
|
|
|
|
|
self.assemble_logical_lines = assemble_logical_lines()
|
|
|
|
if logical_line_transforms is not None:
|
|
|
|
self.logical_line_transforms = logical_line_transforms
|
|
|
|
else:
|
|
|
|
self.logical_line_transforms = [
|
|
|
|
help_end(),
|
|
|
|
escaped_commands(),
|
|
|
|
assign_from_magic(),
|
|
|
|
assign_from_system(),
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
def transforms(self):
|
|
|
|
"Quick access to all transformers."
|
|
|
|
return self.physical_line_transforms + \
|
|
|
|
[self.assemble_logical_lines] + self.logical_line_transforms
|
|
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
def transforms_in_use(self):
|
|
|
|
"""Transformers, excluding logical line transformers if we're in a
|
|
|
|
Python line."""
|
|
|
|
t = self.physical_line_transforms + \
|
|
|
|
[self.assemble_logical_lines] + self.logical_line_transforms
|
2018-03-12 15:11:59 +00:00
|
|
|
return t
|
2018-01-18 14:47:27 +00:00
|
|
|
|
2018-03-14 00:41:05 +00:00
|
|
|
def validQuery(self, text, engine, shell, line=None):
|
2018-01-18 14:47:27 +00:00
|
|
|
"""Return whether a legal query
|
|
|
|
"""
|
2018-03-17 10:38:56 +00:00
|
|
|
if shell and text == shell.os:
|
2018-03-12 15:11:59 +00:00
|
|
|
return True
|
2018-01-18 14:47:27 +00:00
|
|
|
if not line:
|
2018-03-12 15:11:59 +00:00
|
|
|
line = text.rstrip()
|
2018-07-21 01:56:48 +01:00
|
|
|
self.errors = []
|
|
|
|
engine.mgoal(errors(self, line),"user",True)
|
2018-10-15 10:24:30 +01:00
|
|
|
return self.errors == []
|
2018-01-18 14:47:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
def reset(self):
|
|
|
|
"""Reset the input buffer and associated state."""
|
|
|
|
#super(YAPInputSplitter, self).reset()
|
|
|
|
self._buffer_raw[:] = []
|
|
|
|
self.source_raw = ''
|
|
|
|
self.transformer_accumulating = False
|
|
|
|
|
|
|
|
for t in self.transforms:
|
|
|
|
try:
|
|
|
|
t.reset()
|
|
|
|
except SyntaxError:
|
|
|
|
# Nothing that calls reset() expects to handle transformer
|
|
|
|
# errors
|
|
|
|
pass
|
|
|
|
|
|
|
|
def flush_transformers(self):
|
|
|
|
def _flush(transform, outs):
|
|
|
|
"""yield transformed lines
|
|
|
|
|
|
|
|
always strings, never None
|
|
|
|
|
|
|
|
transform: the current transform
|
|
|
|
outs: an iterable of previously transformed inputs.
|
|
|
|
Each may be multiline, which will be passed
|
|
|
|
one line at a time to transform.
|
|
|
|
"""
|
|
|
|
for out in outs:
|
|
|
|
for line in out.splitlines():
|
|
|
|
# push one line at a time
|
|
|
|
tmp = transform.push(line)
|
|
|
|
if tmp is not None:
|
|
|
|
yield tmp
|
|
|
|
|
|
|
|
# reset the transform
|
|
|
|
tmp = transform.reset()
|
|
|
|
if tmp is not None:
|
|
|
|
yield tmp
|
|
|
|
|
|
|
|
out = []
|
|
|
|
|
|
|
|
for t in self.transforms:
|
|
|
|
out = _flush(t, out)
|
|
|
|
|
|
|
|
out = list(out)
|
|
|
|
if out:
|
|
|
|
self._store('\n'.join(out))
|
|
|
|
|
|
|
|
def raw_reset(self):
|
|
|
|
"""Return raw input only and perform a full reset.
|
|
|
|
"""
|
|
|
|
out = self.source_raw
|
|
|
|
self.reset()
|
|
|
|
return out
|
|
|
|
|
|
|
|
def source_reset(self):
|
|
|
|
try:
|
|
|
|
self.flush_transformers()
|
|
|
|
return self.source
|
|
|
|
finally:
|
|
|
|
self.reset()
|
|
|
|
|
|
|
|
def push_accepts_more(self):
|
|
|
|
if self.transformer_accumulating:
|
|
|
|
return True
|
|
|
|
else:
|
2018-03-17 10:38:56 +00:00
|
|
|
return self.validQuery(self.source, engine, self.shell)
|
2018-01-18 14:47:27 +00:00
|
|
|
|
|
|
|
def transform_cell(self, cell):
|
|
|
|
"""Process and translate a cell of input.
|
|
|
|
"""
|
|
|
|
self.reset()
|
|
|
|
try:
|
|
|
|
self.push(cell)
|
|
|
|
self.flush_transformers()
|
|
|
|
return self.source
|
|
|
|
finally:
|
|
|
|
self.reset()
|
|
|
|
|
|
|
|
def push(self, lines):
|
2018-10-19 16:19:53 +01:00
|
|
|
"""Push one or more lines of IPython input.
|
2018-01-18 14:47:27 +00:00
|
|
|
|
|
|
|
This stores the given lines and returns a status code indicating
|
|
|
|
whether the code forms a complete Python block or not, after processing
|
2018-10-19 16:19:53 +01:00
|
|
|
all input lines for special IPython syntax.
|
2018-01-18 14:47:27 +00:00
|
|
|
|
|
|
|
Any exceptions generated in compilation are swallowed, but if an
|
|
|
|
exception was produced, the method returns True.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
lines : string
|
|
|
|
One or more lines of Python input.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
is_complete : boolean
|
|
|
|
True if the current input source (the result of the current input
|
|
|
|
plus prior inputs) forms a complete Python execution block. Note that
|
|
|
|
this value is also stored as a private attribute (_is_complete), so it
|
|
|
|
can be queried at any time.
|
|
|
|
"""
|
|
|
|
|
|
|
|
# We must ensure all input is pure unicode
|
|
|
|
lines = cast_unicode(lines, self.encoding)
|
|
|
|
# ''.splitlines() --> [], but we need to push the empty line to transformers
|
|
|
|
lines_list = lines.splitlines()
|
|
|
|
if not lines_list:
|
|
|
|
lines_list = ['']
|
|
|
|
|
|
|
|
# Store raw source before applying any transformations to it. Note
|
|
|
|
# that this must be done *after* the reset() call that would otherwise
|
|
|
|
# flush the buffer.
|
|
|
|
self._store(lines, self._buffer_raw, 'source_raw')
|
|
|
|
|
|
|
|
transformed_lines_list = []
|
|
|
|
for line in lines_list:
|
|
|
|
transformed = self._transform_line(line)
|
|
|
|
if transformed is not None:
|
|
|
|
transformed_lines_list.append(transformed)
|
|
|
|
if transformed_lines_list:
|
|
|
|
transformed_lines = '\n'.join(transformed_lines_list)
|
2019-02-27 04:23:21 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
else:
|
|
|
|
# Got nothing back from transformers - they must be waiting for
|
|
|
|
# more input.
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _transform_line(self, line):
|
|
|
|
"""Push a line of input code through the various transformers.
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
Returns any output from the transformers, or None if a transformer
|
|
|
|
is accumulating lines.
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
Sets self.transformer_accumulating as a side effect.
|
|
|
|
"""
|
|
|
|
def _accumulating(dbg):
|
|
|
|
#print(dbg)
|
|
|
|
self.transformer_accumulating = True
|
|
|
|
return None
|
|
|
|
|
|
|
|
for transformer in self.physical_line_transforms:
|
|
|
|
line = transformer.push(line)
|
|
|
|
if line is None:
|
|
|
|
return _accumulating(transformer)
|
|
|
|
|
|
|
|
for transformer in self.logical_line_transforms:
|
|
|
|
line = transformer.push(line)
|
|
|
|
if line is None:
|
|
|
|
return _accumulating(transformer)
|
|
|
|
|
|
|
|
|
|
|
|
#print("transformers clear") #debug
|
|
|
|
self.transformer_accumulating = False
|
|
|
|
return line
|
|
|
|
|
|
|
|
|
|
|
|
class YAPCompleter(Completer):
|
|
|
|
|
|
|
|
greedy = Bool(False,
|
|
|
|
help="""Activate greedy completion
|
|
|
|
PENDING DEPRECTION. this is now mostly taken care of with Jedi.
|
|
|
|
|
2018-12-05 09:09:19 +00:00
|
|
|
This will enable completion on elements of lists, results of function calls, etc.,
|
2018-01-18 14:47:27 +00:00
|
|
|
but can be unsafe because the code is actually evaluated on TAB.
|
|
|
|
"""
|
|
|
|
).tag(config=True)
|
|
|
|
|
|
|
|
debug = Bool(default_value=False,
|
|
|
|
help='Enable debug for the Completer. Mostly print extra '
|
|
|
|
'information for experimental jedi integration.') \
|
|
|
|
.tag(config=True)
|
|
|
|
|
|
|
|
backslash_combining_completions = Bool(True,
|
|
|
|
help="Enable unicode completions, e.g. \\alpha<tab> . "
|
|
|
|
"Includes completion of latex commands, unicode names, and expanding "
|
|
|
|
"unicode characters back to latex commands.").tag(config=True)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, namespace=None, global_namespace=None, shell=None, **kwargs):
|
|
|
|
"""Create a new completer for the command line.
|
|
|
|
|
|
|
|
Completer(namespace=ns, global_namespace=ns2) -> completer instance.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
self.shell = shell
|
|
|
|
self.magic_escape = ESC_MAGIC
|
|
|
|
super(Completer, self).__init__(**kwargs)
|
2018-01-05 16:57:38 +00:00
|
|
|
|
|
|
|
def complete(self, text, line=None, cursor_pos=None):
|
|
|
|
"""Return the completed text and a list of completions.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
|
|
|
|
text : string
|
|
|
|
A string of text to be completed on. It can be given as empty and
|
|
|
|
instead a line/position pair are given. In this case, the
|
|
|
|
completer itself will split the line like readline does.
|
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
This is called successively with state == 0, 1, 2, ... until it
|
|
|
|
returns None. The completion should begin with 'text'.
|
|
|
|
|
2018-01-05 16:57:38 +00:00
|
|
|
line : string, optional
|
|
|
|
The complete line that text is part of.
|
|
|
|
|
|
|
|
cursor_pos : int, optional
|
|
|
|
The position of the cursor on the input line.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
text : string
|
|
|
|
The actual text that was completed.
|
|
|
|
|
|
|
|
matches : list
|
|
|
|
A sorted list with all possible completions.
|
|
|
|
|
|
|
|
The optional arguments allow the completion to take more context into
|
|
|
|
account, and are part of the low-level completion API.
|
|
|
|
|
|
|
|
This is a wrapper around the completion mechanism, similar to what
|
|
|
|
readline does at the command line when the TAB key is hit. By
|
|
|
|
exposing it as a method, it can be used by other non-readline
|
|
|
|
environments (such as GUIs) for text completion.
|
|
|
|
|
|
|
|
Simple usage example:
|
|
|
|
|
|
|
|
In [1]: x = 'hello'
|
|
|
|
|
|
|
|
In [2]: _ip.complete('x.l')
|
|
|
|
Out[2]: ('x.l', ['x.ljust', 'x.lower', 'x.lstrip'])
|
|
|
|
"""
|
|
|
|
if not text:
|
|
|
|
text = line[:cursor_pos]
|
2018-01-18 14:47:27 +00:00
|
|
|
return self.completions(text, cursor_pos)
|
|
|
|
|
|
|
|
|
|
|
|
def magic_matches(self, text):
|
|
|
|
"""Match magics"""
|
|
|
|
# Get all shell magics now rather than statically, so magics loaded at
|
|
|
|
# runtime show up too.
|
|
|
|
lsm = self.shell.magics_manager.lsmagic()
|
|
|
|
line_magics = lsm['line']
|
|
|
|
cell_magics = lsm['cell']
|
|
|
|
pre = self.magic_escape
|
|
|
|
pre2 = pre+pre
|
|
|
|
|
|
|
|
explicit_magic = text.startswith(pre)
|
|
|
|
|
|
|
|
# Completion logic:
|
|
|
|
# - user gives %%: only do cell magics
|
|
|
|
# - user gives %: do both line and cell magics
|
|
|
|
# - no prefix: do both
|
|
|
|
# In other words, line magics are skipped if the user gives %% explicitly
|
|
|
|
#
|
|
|
|
# We also exclude magics that match any currently visible names:
|
|
|
|
# https://github.com/ipython/ipython/issues/4877, unless the user has
|
|
|
|
# typed a %:
|
|
|
|
# https://github.com/ipython/ipython/issues/10754
|
|
|
|
bare_text = text.lstrip(pre)
|
|
|
|
global_matches = []
|
|
|
|
if not explicit_magic:
|
|
|
|
def matches(magic):
|
|
|
|
"""
|
|
|
|
Filter magics, in particular remove magics that match
|
|
|
|
a name present in global namespace.
|
|
|
|
"""
|
|
|
|
return ( magic.startswith(bare_text) and
|
|
|
|
magic not in global_matches )
|
|
|
|
else:
|
2018-12-21 20:57:53 +00:00
|
|
|
def matches(magic):
|
2018-01-18 14:47:27 +00:00
|
|
|
return magic.startswith(bare_text)
|
|
|
|
|
|
|
|
comp = [ pre2+m for m in cell_magics if matches(m)]
|
|
|
|
if not text.startswith(pre2):
|
|
|
|
comp += [ pre+m for m in line_magics if matches(m)]
|
|
|
|
|
|
|
|
return comp
|
|
|
|
|
2018-12-05 09:09:19 +00:00
|
|
|
def magic_config_matches(self, text:str) -> List:
|
2018-01-18 14:47:27 +00:00
|
|
|
""" Match class names and attributes for %config magic """
|
|
|
|
texts = text.strip().split()
|
|
|
|
|
|
|
|
if len(texts) > 0 and (texts[0] == 'config' or texts[0] == '%config'):
|
|
|
|
# get all configuration classes
|
|
|
|
classes = sorted(set([ c for c in self.shell.configurables
|
|
|
|
if c.__class__.class_traits(config=True)
|
|
|
|
]), key=lambda x: x.__class__.__name__)
|
|
|
|
classnames = [ c.__class__.__name__ for c in classes ]
|
|
|
|
|
|
|
|
# return all classnames if config or %config is given
|
|
|
|
if len(texts) == 1:
|
|
|
|
return classnames
|
|
|
|
|
|
|
|
# match classname
|
|
|
|
classname_texts = texts[1].split('.')
|
|
|
|
classname = classname_texts[0]
|
|
|
|
classname_matches = [ c for c in classnames
|
|
|
|
if c.startswith(classname) ]
|
|
|
|
|
|
|
|
# return matched classes or the matched class with attributes
|
|
|
|
if texts[1].find('.') < 0:
|
|
|
|
return classname_matches
|
|
|
|
elif len(classname_matches) == 1 and \
|
|
|
|
classname_matches[0] == classname:
|
|
|
|
cls = classes[classnames.index(classname)].__class__
|
|
|
|
help = cls.class_get_help()
|
|
|
|
# strip leading '--' from cl-args:
|
|
|
|
help = re.sub(re.compile(r'^--', re.MULTILINE), '', help)
|
|
|
|
return [ attr.split('=')[0]
|
|
|
|
for attr in help.strip().splitlines()
|
|
|
|
if attr.startswith(texts[1]) ]
|
|
|
|
return []
|
|
|
|
|
|
|
|
|
2018-12-05 09:09:19 +00:00
|
|
|
def magic_color_matches(self, text:str) -> List:
|
2018-01-18 14:47:27 +00:00
|
|
|
""" Match color schemes for %colors magic"""
|
|
|
|
texts = text.split()
|
|
|
|
if text.endswith(' '):
|
|
|
|
# .split() strips off the trailing whitespace. Add '' back
|
|
|
|
# so that: '%colors ' -> ['%colors', '']
|
|
|
|
texts.append('')
|
|
|
|
|
|
|
|
if len(texts) == 2 and (texts[0] == 'colors' or texts[0] == '%colors'):
|
|
|
|
prefix = texts[1]
|
|
|
|
return [ color for color in InspectColors.keys()
|
|
|
|
if color.startswith(prefix) ]
|
|
|
|
return []
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def completions(self, text, offset):
|
|
|
|
"""
|
|
|
|
Returns an iterator over the possible completions
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
.. warning:: Unstable
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
This function is unstable, API may change without warning.
|
|
|
|
It will also raise unless use in proper context manager.
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
Parameters
|
|
|
|
----------
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
text:str
|
|
|
|
Full text of the current input, multi line string.
|
|
|
|
offset:int
|
|
|
|
Integer representing the position of the cursor in ``text``. Offset
|
|
|
|
is 0-based indexed.
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
Yields
|
|
|
|
------
|
|
|
|
:any:`Completion` object
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
|
|
|
|
The cursor on a text can either be seen as being "in between"
|
|
|
|
characters or "On" a character depending on the interface visible to
|
|
|
|
the user. For consistency the cursor being on "in between" characters X
|
|
|
|
and Y is equivalent to the cursor being "on" character Y, that is to say
|
|
|
|
the character the cursor is on is considered as being after the cursor.
|
|
|
|
|
|
|
|
Combining characters may span more that one position in the
|
|
|
|
text.
|
|
|
|
|
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
If ``IPCompleter.debug`` is :any:`True` will yield a ``--jedi/ipython--``
|
|
|
|
fake Completion token to distinguish completion returned by Jedi
|
2018-10-19 16:19:53 +01:00
|
|
|
and usual IPython completion.
|
2018-01-18 14:47:27 +00:00
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
Completions are not completely deduplicated yet. If identical
|
|
|
|
completions are coming from different sources this function does not
|
|
|
|
ensure that each completion object will only be present once.
|
2018-01-05 16:57:38 +00:00
|
|
|
"""
|
2018-10-15 10:24:30 +01:00
|
|
|
if text[0] == '%' and text.find('\n') > offset:
|
|
|
|
magic_res = self.magic_matches(text)
|
|
|
|
return text, magic_res
|
2018-01-18 14:47:27 +00:00
|
|
|
self.matches = []
|
2018-12-21 20:57:53 +00:00
|
|
|
prolog_res = self.shell.engine.mgoal(completions(text, self), "user",True)
|
2018-10-15 10:24:30 +01:00
|
|
|
return text, self.matches
|
2018-11-08 02:33:36 +00:00
|
|
|
|
2018-01-05 16:57:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-10-19 16:19:53 +01:00
|
|
|
class YAPRun(InteractiveShell):
|
2018-01-05 16:57:38 +00:00
|
|
|
"""An enhanced, interactive shell for YAP."""
|
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
def __init__(self, shell):
|
|
|
|
self.shell = shell
|
2018-12-21 20:57:53 +00:00
|
|
|
self.engine = JupyterEngine()
|
2018-03-17 10:38:56 +00:00
|
|
|
global engine
|
2018-12-21 20:57:53 +00:00
|
|
|
engine = self.engine
|
2018-07-27 11:11:04 +01:00
|
|
|
self.errors = []
|
2019-03-19 18:42:17 +00:00
|
|
|
self.q = None
|
2018-03-02 21:18:24 +00:00
|
|
|
self.os = None
|
|
|
|
self.it = None
|
2018-12-12 00:39:17 +00:00
|
|
|
self.port = "None"
|
2018-07-23 17:13:51 +01:00
|
|
|
self.answers = None
|
|
|
|
self.bindings = dicts = []
|
2018-12-21 20:57:53 +00:00
|
|
|
self.shell.engine = self.engine
|
2018-01-18 14:47:27 +00:00
|
|
|
self._get_exc_info = shell._get_exc_info
|
2018-12-21 20:57:53 +00:00
|
|
|
self.iterations = 0
|
2018-01-18 14:47:27 +00:00
|
|
|
|
2018-07-21 23:29:01 +01:00
|
|
|
|
2018-07-23 17:13:51 +01:00
|
|
|
def showtraceback(self, exc_info):
|
|
|
|
try:
|
|
|
|
(etype, value, tb) = e
|
|
|
|
traceback.print_exception(etype, value, tb)
|
|
|
|
except:
|
|
|
|
print(e)
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2018-07-10 23:21:19 +01:00
|
|
|
def syntaxErrors(self, text):
|
2018-01-18 14:47:27 +00:00
|
|
|
"""Return whether a legal query
|
|
|
|
"""
|
|
|
|
if not text:
|
|
|
|
return []
|
2018-03-12 15:11:59 +00:00
|
|
|
if text == self.os:
|
2018-07-21 01:56:48 +01:00
|
|
|
return self.errors
|
|
|
|
self.errors=[]
|
2018-12-21 20:57:53 +00:00
|
|
|
self.engine.mgoal(errors(self,text),"user",True)
|
2018-07-21 01:56:48 +01:00
|
|
|
return self.errors
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2019-04-06 10:07:54 +01:00
|
|
|
def prolog(self, ccell, result):
|
2019-03-19 18:42:17 +00:00
|
|
|
|
2018-01-05 16:57:38 +00:00
|
|
|
#
|
2019-03-19 18:42:17 +00:00
|
|
|
# construct a self.query from a one-line string
|
|
|
|
# self.q is opaque to Python
|
2018-03-12 15:11:59 +00:00
|
|
|
try:
|
2018-06-01 08:37:25 +01:00
|
|
|
# sys.settrace(tracefunc)
|
2019-04-06 10:07:54 +01:00
|
|
|
(program, squery, _, howmany) = ccell
|
2019-03-19 18:42:17 +00:00
|
|
|
if self.q and self.os == (program,squery):
|
2018-06-01 08:37:25 +01:00
|
|
|
howmany += self.iterations
|
|
|
|
else:
|
2019-03-19 18:42:17 +00:00
|
|
|
if self.q:
|
|
|
|
self.q.close()
|
|
|
|
self.q = None
|
|
|
|
self.answers = []
|
|
|
|
result.result = []
|
2018-12-21 20:57:53 +00:00
|
|
|
self.os = (program,squery)
|
2019-03-19 18:42:17 +00:00
|
|
|
pg = jupyter_query(self,program,squery)
|
|
|
|
self.q = Query(self.engine, pg)
|
2019-03-26 09:40:54 +00:00
|
|
|
for v in self.q:
|
2018-03-12 15:11:59 +00:00
|
|
|
self.iterations += 1
|
2019-03-19 18:42:17 +00:00
|
|
|
o = '[ '
|
|
|
|
o += str(self.iterations )
|
|
|
|
o += ' '
|
|
|
|
o += json.dumps(self.q.answer)
|
|
|
|
o += ' ]\n\n'
|
|
|
|
sys.stderr.write( o )
|
|
|
|
self.answers += [self.q.answer]
|
|
|
|
if self.q.port == "exit":
|
|
|
|
break
|
|
|
|
if self.iterations == howmany:
|
|
|
|
break
|
|
|
|
if self.q.port != "answer" and self.iterations == howmany:
|
|
|
|
self.q.close()
|
|
|
|
self.q = None
|
2018-12-21 20:57:53 +00:00
|
|
|
if self.answers:
|
2019-03-19 18:42:17 +00:00
|
|
|
return self.answers
|
2019-02-27 04:23:21 +00:00
|
|
|
else:
|
2019-03-19 18:42:17 +00:00
|
|
|
return None
|
2019-02-27 04:23:21 +00:00
|
|
|
|
2018-12-12 00:39:17 +00:00
|
|
|
|
2018-03-12 15:11:59 +00:00
|
|
|
except Exception as e:
|
2019-03-19 18:42:17 +00:00
|
|
|
sys.stderr.write('Exception '+str(e)+' in query '+ str(self.q)+
|
|
|
|
'\n Answers'+ json.dumps( self.answers)+ '\n')
|
|
|
|
|
2018-03-12 15:11:59 +00:00
|
|
|
has_raised = True
|
2018-12-21 20:57:53 +00:00
|
|
|
return result.result
|
2018-01-05 16:57:38 +00:00
|
|
|
|
|
|
|
|
2018-12-05 09:09:19 +00:00
|
|
|
def _yrun_cell(self, raw_cell, result, store_history=True, silent=False,
|
2018-10-19 16:19:53 +01:00
|
|
|
shell_futures=True):
|
2018-11-08 02:33:36 +00:00
|
|
|
"""Run a complete IPython cell.
|
|
|
|
|
|
|
|
Parameters
|
2018-10-19 16:19:53 +01:00
|
|
|
----------
|
|
|
|
raw_cell : str
|
|
|
|
The code (including IPython code such as
|
|
|
|
%magic functions) to run.
|
|
|
|
store_history : bool
|
|
|
|
If True, the raw and translated cell will be stored in IPython's
|
|
|
|
history. For user code calling back into
|
|
|
|
IPython's machinery, this
|
|
|
|
should be set to False.
|
|
|
|
silent : bool
|
|
|
|
If True, avoid side-effects, such as implicit displayhooks and
|
|
|
|
and logging. silent=True forces store_history=False.
|
|
|
|
shell_futures : bool
|
|
|
|
If True, the code will share future statements with the interactive
|
|
|
|
shell. It will both be affected by previous
|
|
|
|
__future__ imports, and any __future__ imports in the code
|
|
|
|
will affect the shell. If False,
|
2018-01-05 16:57:38 +00:00
|
|
|
__future__ imports are not shared in either direction.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
|
|
|
|
-------
|
2018-12-05 09:09:19 +00:00
|
|
|
`result : :class:`ExecutionResult`
|
2018-11-08 02:33:36 +00:00
|
|
|
"""
|
2018-01-05 16:57:38 +00:00
|
|
|
|
|
|
|
# construct a query from a one-line string
|
|
|
|
# q is opaque to Python
|
|
|
|
# vs is the list of variables
|
|
|
|
# you can print it out, the left-side is the variable name,
|
|
|
|
# the right side wraps a handle to a variable
|
2018-03-19 11:43:14 +00:00
|
|
|
#import pdb; pdb.set_trace()
|
2018-07-21 01:56:48 +01:00
|
|
|
# #pdb.set_trace()
|
2018-12-05 09:09:19 +00:00
|
|
|
# atom match either symbols, or if no symbol exists, strings, In this case # If any of our input transformation (input_transformer_manager or
|
2018-01-05 16:57:38 +00:00
|
|
|
# prefilter_manager) raises an exception, we store it in this variable
|
|
|
|
# so that we can display the error after logging the input and storing
|
|
|
|
# it in the history.
|
|
|
|
preprocessing_exc_tuple = None
|
2018-01-18 14:47:27 +00:00
|
|
|
# try:
|
|
|
|
# # Static input transformations
|
|
|
|
# cell = self.shell.input_transformer_manager.transform_cell(raw_cell)
|
|
|
|
# except SyntaxError:
|
|
|
|
# preprocessing_exc_tuple = self.shell.syntax_error() # sys.exc_info()
|
2018-12-05 09:09:19 +00:00
|
|
|
cell = raw_cell
|
2018-07-27 11:11:04 +01:00
|
|
|
for i in self.errors:
|
2018-01-18 14:47:27 +00:00
|
|
|
try:
|
2018-07-27 11:11:04 +01:00
|
|
|
(_,lin,pos,text) = i
|
2018-10-15 23:59:08 +01:00
|
|
|
e = self.SyntaxError( (self.cell_name, lin, pos, text+'\n'))
|
2018-01-18 14:47:27 +00:00
|
|
|
raise e
|
|
|
|
except SyntaxError:
|
|
|
|
self.shell.showsyntaxerror( )
|
|
|
|
preprocessing_exc_tuple = sys.exc_info()
|
2018-01-05 16:57:38 +00:00
|
|
|
# Store raw and processed history
|
|
|
|
if store_history:
|
2018-01-18 14:47:27 +00:00
|
|
|
self.shell.history_manager.store_inputs(self.shell.execution_count,
|
2018-01-05 16:57:38 +00:00
|
|
|
cell, raw_cell)
|
|
|
|
if not silent:
|
2018-01-18 14:47:27 +00:00
|
|
|
self.shell.logger.log(cell, raw_cell)
|
2018-01-05 16:57:38 +00:00
|
|
|
# # Display the exception if input processing failed.
|
2018-03-19 11:43:14 +00:00
|
|
|
if preprocessing_exc_tuple is not None:
|
|
|
|
self.showtraceback(preprocessing_exc_tuple)
|
2019-04-06 10:07:54 +01:00
|
|
|
if store_history:
|
|
|
|
self.shell.execution_count += 1
|
2018-07-27 11:11:04 +01:00
|
|
|
return self.error_before_exec(preprocessing_exc_tuple[2])
|
2018-01-05 16:57:38 +00:00
|
|
|
|
|
|
|
# Our own compiler remembers the __future__ environment. If we want to
|
|
|
|
# run code with a separate __future__ environment, use the default
|
|
|
|
# compiler
|
2018-01-18 14:47:27 +00:00
|
|
|
# compiler = self.shell.compile if shell_futures else CachingCompiler()
|
2018-07-27 11:11:04 +01:00
|
|
|
self.cell_name = str( self.shell.execution_count)
|
2018-12-05 09:09:19 +00:00
|
|
|
self.shell.displayhook.exec_result= result
|
|
|
|
cell = raw_cell.strip()
|
2019-04-06 10:07:54 +01:00
|
|
|
while cell[0] == '%':
|
2018-01-05 16:57:38 +00:00
|
|
|
if cell[1] == '%':
|
2018-12-05 09:09:19 +00:00
|
|
|
## cell magic
|
|
|
|
txt0 = cell[2:].split(maxsplit = 1, sep = '\n')
|
|
|
|
try:
|
|
|
|
body = txt0[1][1:]
|
|
|
|
magic = txt0[0].strip()
|
|
|
|
except:
|
|
|
|
magic = cell[2:].strip()
|
|
|
|
body = ""
|
2019-04-06 10:07:54 +01:00
|
|
|
linec = False
|
2018-12-05 09:09:19 +00:00
|
|
|
try:
|
|
|
|
[magic,line] = magic.split(maxsplit=1)
|
|
|
|
except:
|
|
|
|
line = ""
|
|
|
|
self.shell.last_execution_succeeded = True
|
2018-12-12 00:39:17 +00:00
|
|
|
result.result = self.shell.run_cell_magic(magic, line, body)
|
2018-12-05 09:09:19 +00:00
|
|
|
return
|
2018-01-05 16:57:38 +00:00
|
|
|
else:
|
2019-04-06 10:07:54 +01:00
|
|
|
linec = True
|
2018-12-05 09:09:19 +00:00
|
|
|
rcell = cell[1:].strip()
|
|
|
|
try:
|
|
|
|
[magic,cell] = rcell.split(maxsplit = 1, sep = '\n')
|
|
|
|
except:
|
|
|
|
magic = rcell.strip()
|
|
|
|
cell = ""
|
|
|
|
try:
|
|
|
|
[magic,line] = magic.split(maxsplit=1)
|
|
|
|
except:
|
|
|
|
line = ""
|
|
|
|
self.shell.last_execution_succeeded = True
|
|
|
|
self.shell.run_line_magic(magic, line)
|
|
|
|
# go execute the body
|
2018-01-05 16:57:38 +00:00
|
|
|
# Give the displayhook a reference to our ExecutionResult so it
|
|
|
|
# can fill in the output value.
|
2018-12-05 09:09:19 +00:00
|
|
|
self.shell.displayhook.exec_result = result
|
2019-04-06 10:07:54 +01:00
|
|
|
ccell = self.prolog_cell(cell)
|
|
|
|
(program,squery,_ ,howmany) = ccell
|
|
|
|
if howmany == 0 and not program:
|
2019-03-26 15:34:42 +00:00
|
|
|
return result
|
|
|
|
if self.syntaxErrors(program+squery+".\n") :
|
2018-12-12 00:39:17 +00:00
|
|
|
result.result = []
|
2019-03-26 09:40:54 +00:00
|
|
|
return result
|
2018-01-05 16:57:38 +00:00
|
|
|
has_raised = False
|
|
|
|
try:
|
2018-12-05 09:09:19 +00:00
|
|
|
builtin_mod.input = input
|
|
|
|
self.shell.input = input
|
2018-12-21 20:57:53 +00:00
|
|
|
self.engine.mgoal(streams(True),"user", True)
|
2019-03-26 09:40:54 +00:00
|
|
|
#create a Trace object, telling it what to ignore, and whether to
|
|
|
|
# do tracing or line-counting or both.
|
|
|
|
# tracer = trace.Trace(
|
|
|
|
# ignoredirs=[sys.prefix, sys.exec_prefix],
|
|
|
|
# trace=1,
|
|
|
|
# count=0)
|
|
|
|
#
|
|
|
|
|
|
|
|
# def f(self, cell, state):
|
|
|
|
# state = self.jupyter_query( cell )
|
2018-06-01 08:37:25 +01:00
|
|
|
|
2018-12-11 12:50:52 +00:00
|
|
|
# run the new command using the given tracer
|
|
|
|
#
|
|
|
|
# tracer.runfunc(f,self,cell,state)
|
2019-04-06 10:07:54 +01:00
|
|
|
answers = self.prolog( ccell, result )
|
2019-03-26 09:40:54 +00:00
|
|
|
# state = tracer.runfunc(hist
|
|
|
|
# er_query( self, cell ) )
|
2018-01-05 16:57:38 +00:00
|
|
|
except Exception as e:
|
|
|
|
has_raised = True
|
2018-07-23 17:13:51 +01:00
|
|
|
try:
|
|
|
|
(etype, value, tb) = e
|
|
|
|
traceback.print_exception(etype, value, tb)
|
|
|
|
except:
|
|
|
|
print(e)
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-01-18 14:47:27 +00:00
|
|
|
self.shell.last_execution_succeeded = not has_raised
|
2018-01-05 16:57:38 +00:00
|
|
|
|
|
|
|
# Reset this so later displayed values do not modify the
|
|
|
|
# ExecutionResult
|
2018-01-18 14:47:27 +00:00
|
|
|
self.shell.displayhook.exec_result = None
|
|
|
|
self.shell.events.trigger('post_execute')
|
2018-01-05 16:57:38 +00:00
|
|
|
if not silent:
|
2018-01-18 14:47:27 +00:00
|
|
|
self.shell.events.trigger('post_run_cell')
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2019-04-06 10:07:54 +01:00
|
|
|
if store_history:
|
|
|
|
# Write output to the database. Does nothing unless
|
|
|
|
# history output logging is enabled.
|
|
|
|
self.shell.history_manager.store_output(self.shell.execution_count)
|
|
|
|
# Each cell is a *single* input, regardless of how many lines it has
|
|
|
|
self.shell.execution_count += 1
|
|
|
|
|
2018-12-21 20:57:53 +00:00
|
|
|
self.engine.mgoal(streams(False),"user", True)
|
2018-12-12 00:39:17 +00:00
|
|
|
return
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2018-03-12 15:11:59 +00:00
|
|
|
|
|
|
|
|
2019-03-26 15:34:42 +00:00
|
|
|
def prolog_cell(self, s):
|
2019-04-06 10:07:54 +01:00
|
|
|
return cell_structure(s)
|
2019-03-27 11:01:55 +00:00
|
|
|
|
2019-04-06 10:07:54 +01:00
|
|
|
def cell_structure(s):
|
2019-03-27 16:31:31 +00:00
|
|
|
"""
|
2019-04-06 10:07:54 +01:00
|
|
|
Trasform a text into program+query. A query is the
|
|
|
|
last line if the last line is non-empty and does not terminate
|
|
|
|
on a dot. You can also finish with
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2019-04-06 10:07:54 +01:00
|
|
|
- `*`: you request all solutions
|
|
|
|
- ';'[N]: you want an answer; optionally you want N answers
|
2018-01-05 16:57:38 +00:00
|
|
|
|
2019-04-06 10:07:54 +01:00
|
|
|
If the line terminates on a `*/` or starts on a `%` we assume the line
|
|
|
|
is a comment.
|
2019-03-27 16:31:31 +00:00
|
|
|
"""
|
|
|
|
try:
|
|
|
|
sl = s.splitlines()
|
|
|
|
l = len(sl)
|
|
|
|
i = 0
|
|
|
|
while i<l:
|
|
|
|
line = sl[-i-1]
|
|
|
|
if line.strip() != '' and line[0] != '':
|
|
|
|
break
|
|
|
|
i+=1
|
|
|
|
if i == l:
|
|
|
|
return ('','','',0)
|
|
|
|
if line[-1] == '.':
|
|
|
|
return (s,'','.',0)
|
|
|
|
query = ''
|
|
|
|
while i<l:
|
|
|
|
line = sl[-i-1]
|
|
|
|
if line.strip() == '':
|
|
|
|
break
|
2019-04-06 10:07:54 +01:00
|
|
|
query = line+'\n\n'+query
|
2019-03-27 16:31:31 +00:00
|
|
|
i+=1
|
2019-04-06 10:07:54 +01:00
|
|
|
reps = 1
|
2019-03-27 16:31:31 +00:00
|
|
|
if query:
|
|
|
|
q = query.strip()
|
2019-04-06 10:07:54 +01:00
|
|
|
c= q.rpartition('*')
|
|
|
|
c2 = c[2].strip()
|
|
|
|
if c[1] != '*' or (c2!='' and not c2.isdecimal()):
|
|
|
|
c = q.rpartition('?')
|
2019-03-27 16:31:31 +00:00
|
|
|
c2 = c[2].strip()
|
2019-04-06 10:07:54 +01:00
|
|
|
if c[1] == '?' and(c2=='' or c2.isdecimal()):
|
|
|
|
c = q.rpartition(';')
|
|
|
|
c2 = c[2].strip()
|
2019-03-27 16:31:31 +00:00
|
|
|
else:
|
2019-04-06 10:07:54 +01:00
|
|
|
c=('','',query)
|
|
|
|
[q,loop,repeats] = c
|
|
|
|
if q:
|
|
|
|
query=q
|
|
|
|
if repeats.strip().isdecimal():
|
|
|
|
reps = int(repeats)
|
|
|
|
sep = sepator
|
2019-04-07 23:09:01 +01:00
|
|
|
elif loop == ';':
|
2019-04-06 10:07:54 +01:00
|
|
|
reps = -1
|
|
|
|
elif loop == '?':
|
|
|
|
reps = 10
|
2019-03-27 16:31:31 +00:00
|
|
|
while i<l:
|
|
|
|
line = sl[-i-1]
|
|
|
|
if line.strip() != '':
|
|
|
|
break
|
|
|
|
i+=1
|
|
|
|
program = ''
|
|
|
|
while i<l:
|
|
|
|
line = sl[-i-1]
|
|
|
|
program = line+'\n'+program
|
|
|
|
i+=1
|
|
|
|
return (program, query, loop, reps)
|
|
|
|
except Exception as e:
|
|
|
|
try:
|
|
|
|
(etype, value, tb) = e
|
|
|
|
traceback.print_exception(etype, value, tb)
|
|
|
|
except:
|
|
|
|
print(e)
|
2018-08-21 14:34:41 +01:00
|
|
|
|