2023-05-23 19:06:25 +02:00
|
|
|
import sys
|
|
|
|
import builtins
|
|
|
|
import traceback
|
|
|
|
from io import TextIOBase
|
|
|
|
from threading import Lock
|
|
|
|
|
2023-06-02 11:13:10 +02:00
|
|
|
from pandemoniummonium._hazmat.conversion cimport (
|
2023-06-02 11:05:45 +02:00
|
|
|
pandemonium_string_to_pyobj,
|
|
|
|
pyobj_to_pandemonium_string,
|
|
|
|
pandemonium_variant_to_pyobj,
|
2023-05-23 19:06:25 +02:00
|
|
|
)
|
2023-06-02 11:13:10 +02:00
|
|
|
from pandemoniummonium._hazmat.gdnative_api_struct cimport (
|
2023-06-02 11:05:45 +02:00
|
|
|
pandemonium_string,
|
|
|
|
pandemonium_string_name,
|
|
|
|
pandemonium_bool,
|
|
|
|
pandemonium_array,
|
|
|
|
pandemonium_pool_string_array,
|
|
|
|
pandemonium_object,
|
|
|
|
pandemonium_variant,
|
|
|
|
pandemonium_variant_call_error,
|
|
|
|
pandemonium_method_rpc_mode,
|
|
|
|
pandemonium_pluginscript_script_data,
|
|
|
|
pandemonium_pluginscript_instance_data,
|
|
|
|
pandemonium_variant_call_error_error,
|
|
|
|
pandemonium_variant_type
|
2023-05-23 19:06:25 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-06-02 11:05:45 +02:00
|
|
|
cpdef inline void pandemonium_print(str pystr):
|
|
|
|
cdef pandemonium_string gdstr
|
|
|
|
pyobj_to_pandemonium_string(pystr, &gdstr)
|
2023-05-23 19:06:25 +02:00
|
|
|
with nogil:
|
2023-06-02 11:05:45 +02:00
|
|
|
gdapi10.pandemonium_print(&gdstr)
|
|
|
|
gdapi10.pandemonium_string_destroy(&gdstr)
|
2023-05-23 19:06:25 +02:00
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
2023-06-02 11:31:19 +02:00
|
|
|
class StdoutStderrCaptureToPandemonium(StdoutStderrCapture):
|
2023-05-23 19:06:25 +02:00
|
|
|
|
|
|
|
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):
|
2023-06-02 11:05:45 +02:00
|
|
|
cdef pandemonium_string gdstr
|
|
|
|
pyobj_to_pandemonium_string(buff, &gdstr)
|
2023-05-23 19:06:25 +02:00
|
|
|
with nogil:
|
2023-06-02 11:05:45 +02:00
|
|
|
gdapi10.pandemonium_print(&gdstr)
|
|
|
|
gdapi10.pandemonium_string_destroy(&gdstr)
|
2023-05-23 19:06:25 +02:00
|
|
|
|
|
|
|
|
|
|
|
cdef _capture_io_streams = None
|
|
|
|
|
|
|
|
|
|
|
|
cdef install_io_streams_capture():
|
|
|
|
global _capture_io_streams
|
|
|
|
assert _capture_io_streams is None
|
2023-06-02 11:31:19 +02:00
|
|
|
_capture_io_streams = StdoutStderrCaptureToPandemonium()
|
2023-05-23 19:06:25 +02:00
|
|
|
_capture_io_streams.install()
|