import sys import builtins import traceback from io import TextIOBase from threading import Lock from godot._hazmat.conversion cimport ( godot_string_to_pyobj, pyobj_to_godot_string, godot_variant_to_pyobj, ) from godot._hazmat.gdnative_api_struct cimport ( godot_string, godot_string_name, godot_bool, godot_array, godot_pool_string_array, godot_object, godot_variant, godot_variant_call_error, godot_method_rpc_mode, godot_pluginscript_script_data, godot_pluginscript_instance_data, godot_variant_call_error_error, godot_variant_type ) cpdef inline void godot_print(str pystr): cdef godot_string gdstr pyobj_to_godot_string(pystr, &gdstr) with nogil: gdapi10.godot_print(&gdstr) gdapi10.godot_string_destroy(&gdstr) class StdinCapture(TextIOBase): def __init__(self): self._enabled = False self._old_stdin = None def install(self): if self._enabled: raise RuntimeError("Already enabled !") self._old_stdin = sys.stdin sys.stdin = self self._enabled = True def remove(self): if not self._enabled: raise RuntimeError("Not enabled !") sys.stdin = self._old_stdin self._enabled = False class StdoutStderrCapture(TextIOBase): def __init__(self): self._enabled = False self._old_stdout = None self._old_stderr = None def install(self): if self._enabled: raise RuntimeError("Already enabled !") self._old_stderr = sys.stderr sys.stderr = self self._old_stdout = sys.stdout sys.stdout = self self._enabled = True # Don't forget to flush the original streams if any (for instance Windows # GUI app without console have sys.__stdout__/__stderr__ set to None) if self._old_stdout is not None: self._old_stdout.flush() if self._old_stdout is not None: self._old_stdout.flush() def remove(self): if not self._enabled: raise RuntimeError("Not enabled !") # # Sanity check, we shouldn't be mixing # if sys.stderr is not self._stderr or sys.stdout is not self._stdout: # raise RuntimeError("sys.stderr/stdout has been patched in our back !") sys.stderr = self._old_stderr sys.stdout = self._old_stdout self._enabled = False class StdoutStderrCaptureToGodot(StdoutStderrCapture): def __init__(self): self.buffer = "" self.callbacks = {} self._enabled = False self._old_stdout = None self._old_stderr = None self._lock = Lock() def write(self, b): with self._lock: self.buffer += b if "\n" in self.buffer: to_print, self.buffer = self.buffer.rsplit("\n", 1) self._write(to_print) def flush(self): with self._lock: if self.buffer: self._write(self.buffer) self.buffer = "" def _write(self, buff): cdef godot_string gdstr pyobj_to_godot_string(buff, &gdstr) with nogil: gdapi10.godot_print(&gdstr) gdapi10.godot_string_destroy(&gdstr) cdef _capture_io_streams = None cdef install_io_streams_capture(): global _capture_io_streams assert _capture_io_streams is None _capture_io_streams = StdoutStderrCaptureToGodot() _capture_io_streams.install()