mirror of
https://github.com/Relintai/gdnative_python.git
synced 2025-01-23 15:27:21 +01:00
130 lines
3.4 KiB
Cython
130 lines
3.4 KiB
Cython
|
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()
|