2023-05-23 19:06:25 +02:00
|
|
|
import sys
|
|
|
|
import ctypes
|
|
|
|
from code import InteractiveConsole
|
|
|
|
from collections import deque
|
|
|
|
from threading import Thread, Lock, Event
|
|
|
|
from queue import SimpleQueue
|
|
|
|
|
2023-06-02 11:31:19 +02:00
|
|
|
from _pandemonium import StdoutStderrCaptureToPandemonium, StdinCapture
|
2023-06-02 11:13:10 +02:00
|
|
|
from pandemonium import exposed, export, ResourceLoader, VBoxContainer
|
2023-05-23 19:06:25 +02:00
|
|
|
|
|
|
|
from .plugin import BASE_RES
|
|
|
|
|
|
|
|
|
|
|
|
FONT = ResourceLoader.load(f"{BASE_RES}/hack_regular.tres")
|
|
|
|
|
|
|
|
|
2023-06-02 11:31:19 +02:00
|
|
|
class StdoutStderrCaptureToBufferAndPassthrough(StdoutStderrCaptureToPandemonium):
|
2023-05-23 19:06:25 +02:00
|
|
|
def __init__(self):
|
|
|
|
super().__init__()
|
|
|
|
self._buffer = ""
|
|
|
|
|
|
|
|
def _write(self, buff):
|
|
|
|
# _write always executed with _lock taken
|
|
|
|
super()._write(buff)
|
|
|
|
self._buffer += buff
|
|
|
|
|
|
|
|
def read_buffer(self):
|
|
|
|
with self._lock:
|
|
|
|
buffer = self._buffer
|
|
|
|
self._buffer = ""
|
|
|
|
return buffer
|
|
|
|
|
|
|
|
|
|
|
|
class StdinCaptureToBuffer(StdinCapture):
|
|
|
|
def __init__(self):
|
|
|
|
super().__init__()
|
|
|
|
self._lock = Lock()
|
|
|
|
self._has_data = Event()
|
|
|
|
self._buffer = ""
|
|
|
|
self._closed = False
|
|
|
|
|
|
|
|
def _read(self, size=-1):
|
|
|
|
if self._closed:
|
|
|
|
raise EOFError
|
|
|
|
|
|
|
|
if size < 0 or size > len(self._buffer):
|
|
|
|
data = self._buffer
|
|
|
|
self._buffer = ""
|
|
|
|
else:
|
|
|
|
data = self._buffer[:size]
|
|
|
|
self._buffer = self._buffer[size:]
|
|
|
|
|
|
|
|
if not self._buffer:
|
|
|
|
self._has_data.clear()
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
def read(self, size=-1):
|
|
|
|
while True:
|
|
|
|
self._has_data.wait()
|
|
|
|
with self._lock:
|
|
|
|
# Check if a concurrent readinto has already processed the data
|
|
|
|
if not self._has_data.is_set():
|
|
|
|
continue
|
|
|
|
|
|
|
|
return self._read(size)
|
|
|
|
|
|
|
|
def readline(size=-1):
|
|
|
|
while True:
|
|
|
|
self._has_data.wait()
|
|
|
|
with self._lock:
|
|
|
|
# Check if a concurrent readinto has already processed the data
|
|
|
|
if not self._has_data.is_set():
|
|
|
|
continue
|
|
|
|
|
|
|
|
if size < 0:
|
|
|
|
size = len(self._buffer)
|
|
|
|
try:
|
|
|
|
size = min(size, self._buffer.index("\n") + 1)
|
|
|
|
except ValueError:
|
|
|
|
# \n not in self._buffer
|
|
|
|
pass
|
|
|
|
return self._read(size)
|
|
|
|
|
|
|
|
def write(self, buffer):
|
|
|
|
if not buffer:
|
|
|
|
return
|
|
|
|
with self._lock:
|
|
|
|
self._has_data.set()
|
|
|
|
self._buffer += buffer
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
self._closed = True
|
|
|
|
# Ensure read is waken up so it can raise EOFError
|
|
|
|
self._has_data.set()
|
|
|
|
|
|
|
|
|
|
|
|
class InteractiveConsoleInREPL(InteractiveConsole):
|
|
|
|
def __init__(self, repl_write, repl_read):
|
|
|
|
super().__init__(locals={"__name__": "__console__", "__doc__": None})
|
|
|
|
# Default write/raw_input relies on stderr/stdin, overwrite them
|
|
|
|
# to only talk with the REPL
|
|
|
|
self.write = repl_write
|
|
|
|
# Note overwritting `InteractiveConsole.raw_input` doesn't prevent
|
|
|
|
# from user code directly calling `input` (for instance when typing
|
|
|
|
# `help()` which makes use of a pager).
|
|
|
|
self.repl_read = repl_read
|
|
|
|
self.thread = None
|
|
|
|
|
|
|
|
def raw_input(self, prompt):
|
|
|
|
data = self.repl_read()
|
|
|
|
# Print the command line in the ouput box, this is needed given
|
|
|
|
# we have a separate input box that is cleared each time
|
|
|
|
# the user hit enter (unlike regular terminal where input and output
|
|
|
|
# are mixed together and enter only jumps to next line)
|
|
|
|
self.write(f"{prompt}{data}")
|
|
|
|
return data
|
|
|
|
|
|
|
|
def start_in_thread(self):
|
|
|
|
assert not self.thread
|
|
|
|
self.thread = Thread(target=self.interact)
|
|
|
|
self.thread.start()
|
|
|
|
|
|
|
|
def send_keyboard_interrupt(self):
|
|
|
|
# Inject a exception in the thread running the interpreter.
|
|
|
|
# This is not 100% perfect given the thread checks for exception only
|
|
|
|
# when it is actually running Python code so we cannot interrupt native
|
|
|
|
# code (for instance calling `time.sleep` cannot be interrupted)
|
|
|
|
ctypes.pythonapi.PyThreadState_SetAsyncExc(
|
|
|
|
self.thread.ident, ctypes.py_object(KeyboardInterrupt)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@exposed(tool=True)
|
|
|
|
class PythonREPL(VBoxContainer):
|
|
|
|
__STREAMS_CAPTURE_INSTALLED = False
|
|
|
|
|
|
|
|
def _enter_tree(self):
|
|
|
|
self.__plugin_instantiated = False
|
|
|
|
self.history = []
|
|
|
|
self.selected_history = 0
|
|
|
|
self.output_box = self.get_node("OutputBox")
|
|
|
|
self.output_box.add_font_override("normal_font", FONT)
|
|
|
|
self.output_box.add_font_override("mono_font", FONT)
|
|
|
|
self.run_button = self.get_node("FooterContainer/RunButton")
|
|
|
|
self.run_button.connect("pressed", self, "execute")
|
|
|
|
self.clear_button = self.get_node("HeaderContainer/ClearButton")
|
|
|
|
self.clear_button.connect("pressed", self, "clear")
|
|
|
|
self.interrupt_button = self.get_node("HeaderContainer/KeyboardInterruptButton")
|
|
|
|
self.interrupt_button.connect("pressed", self, "send_keyboard_interrupt")
|
|
|
|
self.input_box = self.get_node("FooterContainer/InputBox")
|
|
|
|
self.input_box.connect("text_entered", self, "execute")
|
|
|
|
|
|
|
|
# Hijack stdout/stderr/stdin streams
|
|
|
|
self.stdout_stderr_capture = StdoutStderrCaptureToBufferAndPassthrough()
|
|
|
|
self.stdin_capture = StdinCaptureToBuffer()
|
|
|
|
# Only overwrite streams if the scene has been created by the
|
|
|
|
# pythonscript_repl plugin. This avoid concurrent streams patching
|
|
|
|
# when the scene is opened from the editor (typically when we want
|
|
|
|
# to edit the repl GUI)
|
|
|
|
# TODO: find a way to differentiate plugin instantiated from other
|
|
|
|
# instantiations instead of relying on "first instantiated is plugin"
|
|
|
|
if not PythonREPL.__STREAMS_CAPTURE_INSTALLED:
|
|
|
|
PythonREPL.__STREAMS_CAPTURE_INSTALLED = True
|
|
|
|
self.__plugin_instantiated = True
|
|
|
|
self.stdout_stderr_capture.install()
|
|
|
|
self.stdin_capture.install()
|
|
|
|
|
|
|
|
# Finally start the Python interpreter, it must be running it in own
|
|
|
|
# thread given it does blocking reads on stdin
|
|
|
|
self.interpreter = InteractiveConsoleInREPL(
|
|
|
|
repl_write=self.write, repl_read=self.stdin_capture.read
|
|
|
|
)
|
|
|
|
self.interpreter.start_in_thread()
|
|
|
|
|
|
|
|
def _exit_tree(self):
|
|
|
|
# Closing our custom stdin stream should make `InteractiveConsole.interact`
|
|
|
|
# return, hence finishing the interpreter thread
|
|
|
|
self.stdin_capture.close()
|
|
|
|
self.interpreter.thread.join()
|
|
|
|
|
|
|
|
# Our custom stream capture must be removed before this node is destroyed,
|
|
|
|
# otherwise segfault will occur on next print !
|
|
|
|
if self.__plugin_instantiated:
|
|
|
|
PythonREPL.__STREAMS_CAPTURE_INSTALLED = False
|
|
|
|
self.stdout_stderr_capture.remove()
|
|
|
|
self.stdin_capture.remove()
|
|
|
|
|
|
|
|
def write(self, buffer):
|
|
|
|
for line in buffer.splitlines():
|
|
|
|
self.output_box.push_mono()
|
|
|
|
self.output_box.add_text(line)
|
|
|
|
self.output_box.newline()
|
|
|
|
self.output_box.pop()
|
|
|
|
|
|
|
|
def _process(self, delta):
|
|
|
|
if not hasattr(self, "stdout_stderr_capture"):
|
|
|
|
return
|
|
|
|
# Display new lines
|
|
|
|
self.write(self.stdout_stderr_capture.read_buffer())
|
|
|
|
|
|
|
|
def remove_last_line(self):
|
|
|
|
self.output_box.remove_line(self.output_box.get_line_count() - 2)
|
|
|
|
self.output_box.scroll_to_line(self.output_box.get_line_count() - 1)
|
|
|
|
|
|
|
|
def execute(self, *args, **kwargs):
|
|
|
|
string = str(self.input_box.get_text())
|
|
|
|
# Avoid adding multiple repeated entries to the command history
|
|
|
|
if not (len(self.history) > 0 and self.history[-1] == string):
|
|
|
|
self.history.append(string)
|
|
|
|
self.selected_history = 0
|
|
|
|
self.input_box.clear()
|
|
|
|
# Send the line into stdin and let the interpret do the rest
|
|
|
|
self.stdin_capture.write(string + "\n")
|
|
|
|
|
|
|
|
def up_pressed(self):
|
|
|
|
if len(self.history) >= abs(self.selected_history - 1):
|
|
|
|
self.selected_history -= 1
|
|
|
|
self.input_box.clear()
|
|
|
|
val = str(self.history[self.selected_history])
|
|
|
|
self.input_box.set_text(val)
|
|
|
|
self.input_box.set_cursor_position(len(val))
|
|
|
|
self.input_box.grab_focus()
|
|
|
|
|
|
|
|
def down_pressed(self):
|
|
|
|
if self.selected_history + 1 == 0:
|
|
|
|
self.selected_history += 1
|
|
|
|
self.input_box.clear()
|
|
|
|
elif self.selected_history + 1 < 0:
|
|
|
|
self.selected_history += 1
|
|
|
|
self.input_box.clear()
|
|
|
|
val = str(self.history[self.selected_history])
|
|
|
|
self.input_box.set_text(val)
|
|
|
|
self.input_box.set_cursor_position(len(val))
|
|
|
|
self.input_box.grab_focus()
|
|
|
|
|
|
|
|
def clear(self):
|
|
|
|
self.output_box.clear()
|
|
|
|
|
|
|
|
def send_keyboard_interrupt(self):
|
|
|
|
self.interpreter.send_keyboard_interrupt()
|