| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  | """Subinterpreters High Level Module.""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import threading | 
					
						
							|  |  |  | import weakref | 
					
						
							|  |  |  | import _xxsubinterpreters as _interpreters | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # aliases: | 
					
						
							|  |  |  | from _xxsubinterpreters import ( | 
					
						
							| 
									
										
										
										
											2024-02-28 16:08:08 -07:00
										 |  |  |     InterpreterError, InterpreterNotFoundError, NotShareableError, | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |     is_shareable, | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __all__ = [ | 
					
						
							|  |  |  |     'get_current', 'get_main', 'create', 'list_all', 'is_shareable', | 
					
						
							|  |  |  |     'Interpreter', | 
					
						
							| 
									
										
										
										
											2024-02-28 16:08:08 -07:00
										 |  |  |     'InterpreterError', 'InterpreterNotFoundError', 'ExecutionFailed', | 
					
						
							|  |  |  |     'NotShareableError', | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |     'create_queue', 'Queue', 'QueueEmpty', 'QueueFull', | 
					
						
							|  |  |  | ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _queuemod = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def __getattr__(name): | 
					
						
							|  |  |  |     if name in ('Queue', 'QueueEmpty', 'QueueFull', 'create_queue'): | 
					
						
							|  |  |  |         global create_queue, Queue, QueueEmpty, QueueFull | 
					
						
							|  |  |  |         ns = globals() | 
					
						
							|  |  |  |         from .queues import ( | 
					
						
							|  |  |  |             create as create_queue, | 
					
						
							|  |  |  |             Queue, QueueEmpty, QueueFull, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         return ns[name] | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         raise AttributeError(name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-12 17:00:54 -07:00
										 |  |  | _EXEC_FAILURE_STR = """
 | 
					
						
							|  |  |  | {superstr} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Uncaught in the interpreter: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | {formatted} | 
					
						
							|  |  |  | """.strip()
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 16:08:08 -07:00
										 |  |  | class ExecutionFailed(RuntimeError): | 
					
						
							|  |  |  |     """An unhandled exception happened during execution.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This is raised from Interpreter.exec() and Interpreter.call(). | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, excinfo): | 
					
						
							|  |  |  |         msg = excinfo.formatted | 
					
						
							|  |  |  |         if not msg: | 
					
						
							| 
									
										
										
										
											2023-12-12 17:00:54 -07:00
										 |  |  |             if excinfo.type and excinfo.msg: | 
					
						
							|  |  |  |                 msg = f'{excinfo.type.__name__}: {excinfo.msg}' | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2023-12-12 17:00:54 -07:00
										 |  |  |                 msg = excinfo.type.__name__ or excinfo.msg | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |         super().__init__(msg) | 
					
						
							| 
									
										
										
										
											2023-12-12 17:00:54 -07:00
										 |  |  |         self.excinfo = excinfo | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __str__(self): | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2023-12-12 17:31:30 -07:00
										 |  |  |             formatted = self.excinfo.errdisplay | 
					
						
							| 
									
										
										
										
											2023-12-12 17:00:54 -07:00
										 |  |  |         except Exception: | 
					
						
							|  |  |  |             return super().__str__() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return _EXEC_FAILURE_STR.format( | 
					
						
							|  |  |  |                 superstr=super().__str__(), | 
					
						
							|  |  |  |                 formatted=formatted, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def create(): | 
					
						
							|  |  |  |     """Return a new (idle) Python interpreter.""" | 
					
						
							| 
									
										
										
										
											2024-04-02 17:16:50 -06:00
										 |  |  |     id = _interpreters.create(reqrefs=True) | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |     return Interpreter(id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def list_all(): | 
					
						
							|  |  |  |     """Return all existing interpreters.""" | 
					
						
							|  |  |  |     return [Interpreter(id) for id in _interpreters.list_all()] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_current(): | 
					
						
							|  |  |  |     """Return the currently running interpreter.""" | 
					
						
							|  |  |  |     id = _interpreters.get_current() | 
					
						
							|  |  |  |     return Interpreter(id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_main(): | 
					
						
							|  |  |  |     """Return the main interpreter.""" | 
					
						
							|  |  |  |     id = _interpreters.get_main() | 
					
						
							|  |  |  |     return Interpreter(id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _known = weakref.WeakValueDictionary() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Interpreter: | 
					
						
							|  |  |  |     """A single Python interpreter.""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __new__(cls, id, /): | 
					
						
							|  |  |  |         # There is only one instance for any given ID. | 
					
						
							|  |  |  |         if not isinstance(id, int): | 
					
						
							|  |  |  |             raise TypeError(f'id must be an int, got {id!r}') | 
					
						
							|  |  |  |         id = int(id) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             self = _known[id] | 
					
						
							|  |  |  |             assert hasattr(self, '_ownsref') | 
					
						
							|  |  |  |         except KeyError: | 
					
						
							|  |  |  |             # This may raise InterpreterNotFoundError: | 
					
						
							| 
									
										
										
										
											2024-04-02 17:16:50 -06:00
										 |  |  |             _interpreters.incref(id) | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 self = super().__new__(cls) | 
					
						
							|  |  |  |                 self._id = id | 
					
						
							|  |  |  |                 self._ownsref = True | 
					
						
							|  |  |  |             except BaseException: | 
					
						
							| 
									
										
										
										
											2024-04-02 17:16:50 -06:00
										 |  |  |                 _interpreters.decref(id) | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             _known[id] = self | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return f'{type(self).__name__}({self.id})' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __hash__(self): | 
					
						
							|  |  |  |         return hash(self._id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __del__(self): | 
					
						
							|  |  |  |         self._decref() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-05 08:54:46 -07:00
										 |  |  |     # for pickling: | 
					
						
							|  |  |  |     def __getnewargs__(self): | 
					
						
							|  |  |  |         return (self._id,) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # for pickling: | 
					
						
							|  |  |  |     def __getstate__(self): | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |     def _decref(self): | 
					
						
							|  |  |  |         if not self._ownsref: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         self._ownsref = False | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2024-04-02 17:16:50 -06:00
										 |  |  |             _interpreters.decref(self.id) | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |         except InterpreterNotFoundError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def id(self): | 
					
						
							|  |  |  |         return self._id | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_running(self): | 
					
						
							|  |  |  |         """Return whether or not the identified interpreter is running.""" | 
					
						
							|  |  |  |         return _interpreters.is_running(self._id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def close(self): | 
					
						
							|  |  |  |         """Finalize and destroy the interpreter.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Attempting to destroy the current interpreter results | 
					
						
							|  |  |  |         in a RuntimeError. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return _interpreters.destroy(self._id) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-12 11:06:06 -07:00
										 |  |  |     def prepare_main(self, ns=None, /, **kwargs): | 
					
						
							|  |  |  |         """Bind the given values into the interpreter's __main__.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The values must be shareable. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         ns = dict(ns, **kwargs) if ns is not None else kwargs | 
					
						
							|  |  |  |         _interpreters.set___main___attrs(self._id, ns) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 16:08:08 -07:00
										 |  |  |     def exec(self, code, /): | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |         """Run the given source code in the interpreter.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This is essentially the same as calling the builtin "exec" | 
					
						
							|  |  |  |         with this interpreter, using the __dict__ of its __main__ | 
					
						
							|  |  |  |         module as both globals and locals. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         There is no return value. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 16:08:08 -07:00
										 |  |  |         If the code raises an unhandled exception then an ExecutionFailed | 
					
						
							|  |  |  |         exception is raised, which summarizes the unhandled exception. | 
					
						
							|  |  |  |         The actual exception is discarded because objects cannot be | 
					
						
							|  |  |  |         shared between interpreters. | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         This blocks the current Python thread until done.  During | 
					
						
							|  |  |  |         that time, the previous interpreter is allowed to run | 
					
						
							|  |  |  |         in other threads. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2023-12-12 11:06:06 -07:00
										 |  |  |         excinfo = _interpreters.exec(self._id, code) | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |         if excinfo is not None: | 
					
						
							| 
									
										
										
										
											2024-02-28 16:08:08 -07:00
										 |  |  |             raise ExecutionFailed(excinfo) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def call(self, callable, /): | 
					
						
							|  |  |  |         """Call the object in the interpreter with given args/kwargs.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Only functions that take no arguments and have no closure | 
					
						
							|  |  |  |         are supported. | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-28 16:08:08 -07:00
										 |  |  |         The return value is discarded. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         If the callable raises an exception then the error display | 
					
						
							|  |  |  |         (including full traceback) is send back between the interpreters | 
					
						
							|  |  |  |         and an ExecutionFailed exception is raised, much like what | 
					
						
							|  |  |  |         happens with Interpreter.exec(). | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # XXX Support args and kwargs. | 
					
						
							|  |  |  |         # XXX Support arbitrary callables. | 
					
						
							|  |  |  |         # XXX Support returning the return value (e.g. via pickle). | 
					
						
							|  |  |  |         excinfo = _interpreters.call(self._id, callable) | 
					
						
							|  |  |  |         if excinfo is not None: | 
					
						
							|  |  |  |             raise ExecutionFailed(excinfo) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def call_in_thread(self, callable, /): | 
					
						
							|  |  |  |         """Return a new thread that calls the object in the interpreter.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The return value and any raised exception are discarded. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |         def task(): | 
					
						
							| 
									
										
										
										
											2024-02-28 16:08:08 -07:00
										 |  |  |             self.call(callable) | 
					
						
							| 
									
										
										
										
											2023-12-12 08:24:31 -07:00
										 |  |  |         t = threading.Thread(target=task) | 
					
						
							|  |  |  |         t.start() | 
					
						
							|  |  |  |         return t |