| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | """Support for remote Python debugging.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Some ASCII art to describe the structure: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        IN PYTHON SUBPROCESS          #             IN IDLE PROCESS | 
					
						
							|  |  |  |                                      # | 
					
						
							|  |  |  |                                      #        oid='gui_adapter' | 
					
						
							|  |  |  |                  +----------+        #       +------------+          +-----+ | 
					
						
							|  |  |  |                  | GUIProxy |--remote#call-->| GUIAdapter |--calls-->| GUI | | 
					
						
							|  |  |  | +-----+--calls-->+----------+        #       +------------+          +-----+ | 
					
						
							|  |  |  | | Idb |                               #                             / | 
					
						
							|  |  |  | +-----+<-calls--+------------+         #      +----------+<--calls-/ | 
					
						
							|  |  |  |                 | IdbAdapter |<--remote#call--| IdbProxy | | 
					
						
							|  |  |  |                 +------------+         #      +----------+ | 
					
						
							|  |  |  |                 oid='idb_adapter'      # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The purpose of the Proxy and Adapter classes is to translate certain | 
					
						
							|  |  |  | arguments and return values that cannot be transported through the RPC | 
					
						
							|  |  |  | barrier, in particular frame and traceback objects. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											2021-01-10 01:59:47 -05:00
										 |  |  | import reprlib | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  | import types | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  | from idlelib import debugger | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  | debugging = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-09-05 02:31:20 +00:00
										 |  |  | idb_adap_oid = "idb_adapter" | 
					
						
							|  |  |  | gui_adap_oid = "gui_adapter" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | #======================================= | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # In the PYTHON subprocess: | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | frametable = {} | 
					
						
							|  |  |  | dicttable = {} | 
					
						
							|  |  |  | codetable = {} | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  | tracebacktable = {} | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def wrap_frame(frame): | 
					
						
							|  |  |  |     fid = id(frame) | 
					
						
							|  |  |  |     frametable[fid] = frame | 
					
						
							|  |  |  |     return fid | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def wrap_info(info): | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |     "replace info[2], a traceback instance, by its ID" | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |     if info is None: | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |         traceback = info[2] | 
					
						
							|  |  |  |         assert isinstance(traceback, types.TracebackType) | 
					
						
							|  |  |  |         traceback_id = id(traceback) | 
					
						
							|  |  |  |         tracebacktable[traceback_id] = traceback | 
					
						
							|  |  |  |         modified_info = (info[0], info[1], traceback_id) | 
					
						
							|  |  |  |         return modified_info | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class GUIProxy: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  |     def __init__(self, conn, gui_adap_oid): | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         self.conn = conn | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  |         self.oid = gui_adap_oid | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def interaction(self, message, frame, info=None): | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  |         # calls rpc.SocketIO.remotecall() via run.MyHandler instance | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |         # pass frame and traceback object IDs instead of the objects themselves | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         self.conn.remotecall(self.oid, "interaction", | 
					
						
							|  |  |  |                              (message, wrap_frame(frame), wrap_info(info)), | 
					
						
							|  |  |  |                              {}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class IdbAdapter: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, idb): | 
					
						
							|  |  |  |         self.idb = idb | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  |     #----------called by an IdbProxy---------- | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |     def set_step(self): | 
					
						
							|  |  |  |         self.idb.set_step() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_quit(self): | 
					
						
							|  |  |  |         self.idb.set_quit() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_continue(self): | 
					
						
							|  |  |  |         self.idb.set_continue() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_next(self, fid): | 
					
						
							|  |  |  |         frame = frametable[fid] | 
					
						
							|  |  |  |         self.idb.set_next(frame) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_return(self, fid): | 
					
						
							|  |  |  |         frame = frametable[fid] | 
					
						
							|  |  |  |         self.idb.set_return(frame) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_stack(self, fid, tbid): | 
					
						
							|  |  |  |         frame = frametable[fid] | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |         if tbid is None: | 
					
						
							|  |  |  |             tb = None | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             tb = tracebacktable[tbid] | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         stack, i = self.idb.get_stack(frame, tb) | 
					
						
							| 
									
										
										
										
											2015-05-15 23:03:17 -04:00
										 |  |  |         stack = [(wrap_frame(frame2), k) for frame2, k in stack] | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         return stack, i | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def run(self, cmd): | 
					
						
							|  |  |  |         import __main__ | 
					
						
							|  |  |  |         self.idb.run(cmd, __main__.__dict__) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  |     def set_break(self, filename, lineno): | 
					
						
							|  |  |  |         msg = self.idb.set_break(filename, lineno) | 
					
						
							|  |  |  |         return msg | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clear_break(self, filename, lineno): | 
					
						
							|  |  |  |         msg = self.idb.clear_break(filename, lineno) | 
					
						
							| 
									
										
										
										
											2002-09-05 02:31:20 +00:00
										 |  |  |         return msg | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-24 17:03:37 +00:00
										 |  |  |     def clear_all_file_breaks(self, filename): | 
					
						
							|  |  |  |         msg = self.idb.clear_all_file_breaks(filename) | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |         return msg | 
					
						
							| 
									
										
										
										
											2002-12-31 16:03:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  |     #----------called by a FrameProxy---------- | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |     def frame_attr(self, fid, name): | 
					
						
							|  |  |  |         frame = frametable[fid] | 
					
						
							|  |  |  |         return getattr(frame, name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def frame_globals(self, fid): | 
					
						
							|  |  |  |         frame = frametable[fid] | 
					
						
							| 
									
										
										
										
											2024-01-17 23:39:12 -05:00
										 |  |  |         gdict = frame.f_globals | 
					
						
							|  |  |  |         did = id(gdict) | 
					
						
							|  |  |  |         dicttable[did] = gdict | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         return did | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def frame_locals(self, fid): | 
					
						
							|  |  |  |         frame = frametable[fid] | 
					
						
							| 
									
										
										
										
											2024-01-17 23:39:12 -05:00
										 |  |  |         ldict = frame.f_locals | 
					
						
							|  |  |  |         did = id(ldict) | 
					
						
							|  |  |  |         dicttable[did] = ldict | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         return did | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def frame_code(self, fid): | 
					
						
							|  |  |  |         frame = frametable[fid] | 
					
						
							|  |  |  |         code = frame.f_code | 
					
						
							|  |  |  |         cid = id(code) | 
					
						
							|  |  |  |         codetable[cid] = code | 
					
						
							|  |  |  |         return cid | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  |     #----------called by a CodeProxy---------- | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |     def code_name(self, cid): | 
					
						
							|  |  |  |         code = codetable[cid] | 
					
						
							|  |  |  |         return code.co_name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def code_filename(self, cid): | 
					
						
							|  |  |  |         code = codetable[cid] | 
					
						
							|  |  |  |         return code.co_filename | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  |     #----------called by a DictProxy---------- | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |     def dict_keys(self, did): | 
					
						
							| 
									
										
										
										
											2018-12-06 22:36:55 +02:00
										 |  |  |         raise NotImplementedError("dict_keys not public or pickleable") | 
					
						
							| 
									
										
										
										
											2024-01-17 23:39:12 -05:00
										 |  |  | ##         return dicttable[did].keys() | 
					
						
							| 
									
										
										
										
											2007-08-29 18:44:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-17 23:39:12 -05:00
										 |  |  |     ### Needed until dict_keys type is finished and pickleable. | 
					
						
							|  |  |  |     # xxx finished. pickleable? | 
					
						
							| 
									
										
										
										
											2007-08-29 18:44:24 +00:00
										 |  |  |     ### Will probably need to extend rpc.py:SocketIO._proxify at that time. | 
					
						
							|  |  |  |     def dict_keys_list(self, did): | 
					
						
							| 
									
										
										
										
											2024-01-17 23:39:12 -05:00
										 |  |  |         return list(dicttable[did].keys()) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def dict_item(self, did, key): | 
					
						
							| 
									
										
										
										
											2024-01-17 23:39:12 -05:00
										 |  |  |         value = dicttable[did][key] | 
					
						
							|  |  |  |         return reprlib.repr(value) # Can't pickle module 'builtins'. | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | #----------end class IdbAdapter---------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  | def start_debugger(rpchandler, gui_adap_oid): | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  |     """Start the debugger and its RPC link in the Python subprocess
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Start the subprocess side of the split debugger and set up that side of the | 
					
						
							| 
									
										
										
										
											2002-06-24 17:03:37 +00:00
										 |  |  |     RPC link by instantiating the GUIProxy, Idb debugger, and IdbAdapter | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  |     objects and linking them together.  Register the IdbAdapter with the | 
					
						
							|  |  |  |     RPCServer to handle RPC requests from the split debugger GUI via the | 
					
						
							|  |  |  |     IdbProxy. | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  |     gui_proxy = GUIProxy(rpchandler, gui_adap_oid) | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  |     idb = debugger.Idb(gui_proxy) | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  |     idb_adap = IdbAdapter(idb) | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  |     rpchandler.register(idb_adap_oid, idb_adap) | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  |     return idb_adap_oid | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #======================================= | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # In the IDLE process: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class FrameProxy: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, conn, fid): | 
					
						
							|  |  |  |         self._conn = conn | 
					
						
							|  |  |  |         self._fid = fid | 
					
						
							|  |  |  |         self._oid = "idb_adapter" | 
					
						
							|  |  |  |         self._dictcache = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __getattr__(self, name): | 
					
						
							|  |  |  |         if name[:1] == "_": | 
					
						
							| 
									
										
										
										
											2007-08-23 01:06:15 +00:00
										 |  |  |             raise AttributeError(name) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         if name == "f_code": | 
					
						
							|  |  |  |             return self._get_f_code() | 
					
						
							|  |  |  |         if name == "f_globals": | 
					
						
							|  |  |  |             return self._get_f_globals() | 
					
						
							|  |  |  |         if name == "f_locals": | 
					
						
							|  |  |  |             return self._get_f_locals() | 
					
						
							|  |  |  |         return self._conn.remotecall(self._oid, "frame_attr", | 
					
						
							|  |  |  |                                      (self._fid, name), {}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_f_code(self): | 
					
						
							|  |  |  |         cid = self._conn.remotecall(self._oid, "frame_code", (self._fid,), {}) | 
					
						
							|  |  |  |         return CodeProxy(self._conn, self._oid, cid) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_f_globals(self): | 
					
						
							|  |  |  |         did = self._conn.remotecall(self._oid, "frame_globals", | 
					
						
							|  |  |  |                                     (self._fid,), {}) | 
					
						
							|  |  |  |         return self._get_dict_proxy(did) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_f_locals(self): | 
					
						
							|  |  |  |         did = self._conn.remotecall(self._oid, "frame_locals", | 
					
						
							|  |  |  |                                     (self._fid,), {}) | 
					
						
							|  |  |  |         return self._get_dict_proxy(did) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_dict_proxy(self, did): | 
					
						
							| 
									
										
										
										
											2006-08-22 15:45:46 +00:00
										 |  |  |         if did in self._dictcache: | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |             return self._dictcache[did] | 
					
						
							|  |  |  |         dp = DictProxy(self._conn, self._oid, did) | 
					
						
							|  |  |  |         self._dictcache[did] = dp | 
					
						
							|  |  |  |         return dp | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | class CodeProxy: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, conn, oid, cid): | 
					
						
							|  |  |  |         self._conn = conn | 
					
						
							|  |  |  |         self._oid = oid | 
					
						
							|  |  |  |         self._cid = cid | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __getattr__(self, name): | 
					
						
							|  |  |  |         if name == "co_name": | 
					
						
							|  |  |  |             return self._conn.remotecall(self._oid, "code_name", | 
					
						
							|  |  |  |                                          (self._cid,), {}) | 
					
						
							|  |  |  |         if name == "co_filename": | 
					
						
							|  |  |  |             return self._conn.remotecall(self._oid, "code_filename", | 
					
						
							|  |  |  |                                          (self._cid,), {}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | class DictProxy: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, conn, oid, did): | 
					
						
							|  |  |  |         self._conn = conn | 
					
						
							|  |  |  |         self._oid = oid | 
					
						
							|  |  |  |         self._did = did | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-29 18:44:24 +00:00
										 |  |  | ##    def keys(self): | 
					
						
							|  |  |  | ##        return self._conn.remotecall(self._oid, "dict_keys", (self._did,), {}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # 'temporary' until dict_keys is a pickleable built-in type | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |     def keys(self): | 
					
						
							| 
									
										
										
										
											2007-08-29 18:44:24 +00:00
										 |  |  |         return self._conn.remotecall(self._oid, | 
					
						
							|  |  |  |                                      "dict_keys_list", (self._did,), {}) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __getitem__(self, key): | 
					
						
							|  |  |  |         return self._conn.remotecall(self._oid, "dict_item", | 
					
						
							|  |  |  |                                      (self._did, key), {}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __getattr__(self, name): | 
					
						
							| 
									
										
										
										
											2007-08-29 18:44:24 +00:00
										 |  |  |         ##print("*** Failed DictProxy.__getattr__:", name) | 
					
						
							| 
									
										
										
										
											2007-08-23 01:06:15 +00:00
										 |  |  |         raise AttributeError(name) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  | class GUIAdapter: | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, conn, gui): | 
					
						
							|  |  |  |         self.conn = conn | 
					
						
							|  |  |  |         self.gui = gui | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |     def interaction(self, message, fid, modified_info): | 
					
						
							| 
									
										
										
										
											2007-08-29 18:44:24 +00:00
										 |  |  |         ##print("*** Interaction: (%s, %s, %s)" % (message, fid, modified_info)) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         frame = FrameProxy(self.conn, fid) | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |         self.gui.interaction(message, frame, modified_info) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | class IdbProxy: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-01-25 21:33:40 +00:00
										 |  |  |     def __init__(self, conn, shell, oid): | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         self.oid = oid | 
					
						
							|  |  |  |         self.conn = conn | 
					
						
							| 
									
										
										
										
											2003-01-25 21:33:40 +00:00
										 |  |  |         self.shell = shell | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-01 11:00:15 +03:00
										 |  |  |     def call(self, methodname, /, *args, **kwargs): | 
					
						
							| 
									
										
										
										
											2007-08-29 18:44:24 +00:00
										 |  |  |         ##print("*** IdbProxy.call %s %s %s" % (methodname, args, kwargs)) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         value = self.conn.remotecall(self.oid, methodname, args, kwargs) | 
					
						
							| 
									
										
										
										
											2007-08-29 18:44:24 +00:00
										 |  |  |         ##print("*** IdbProxy.call %s returns %r" % (methodname, value)) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         return value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def run(self, cmd, locals): | 
					
						
							|  |  |  |         # Ignores locals on purpose! | 
					
						
							| 
									
										
										
										
											2003-05-12 02:33:47 +00:00
										 |  |  |         seq = self.conn.asyncqueue(self.oid, "run", (cmd,), {}) | 
					
						
							| 
									
										
										
										
											2003-01-25 21:33:40 +00:00
										 |  |  |         self.shell.interp.active_seq = seq | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |     def get_stack(self, frame, tbid): | 
					
						
							|  |  |  |         # passing frame and traceback IDs, not the objects themselves | 
					
						
							|  |  |  |         stack, i = self.call("get_stack", frame._fid, tbid) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |         stack = [(FrameProxy(self.conn, fid), k) for fid, k in stack] | 
					
						
							|  |  |  |         return stack, i | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_continue(self): | 
					
						
							|  |  |  |         self.call("set_continue") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_step(self): | 
					
						
							|  |  |  |         self.call("set_step") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_next(self, frame): | 
					
						
							|  |  |  |         self.call("set_next", frame._fid) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_return(self, frame): | 
					
						
							|  |  |  |         self.call("set_return", frame._fid) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_quit(self): | 
					
						
							|  |  |  |         self.call("set_quit") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  |     def set_break(self, filename, lineno): | 
					
						
							|  |  |  |         msg = self.call("set_break", filename, lineno) | 
					
						
							|  |  |  |         return msg | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clear_break(self, filename, lineno): | 
					
						
							|  |  |  |         msg = self.call("clear_break", filename, lineno) | 
					
						
							| 
									
										
										
										
											2002-09-05 02:31:20 +00:00
										 |  |  |         return msg | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-24 17:03:37 +00:00
										 |  |  |     def clear_all_file_breaks(self, filename): | 
					
						
							|  |  |  |         msg = self.call("clear_all_file_breaks", filename) | 
					
						
							| 
									
										
										
										
											2002-07-03 03:55:43 +00:00
										 |  |  |         return msg | 
					
						
							| 
									
										
										
										
											2002-06-24 17:03:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  | def start_remote_debugger(rpcclt, pyshell): | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  |     """Start the subprocess debugger, initialize the debugger GUI and RPC link
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-20 04:01:47 +00:00
										 |  |  |     Request the RPCServer start the Python subprocess debugger and link.  Set | 
					
						
							|  |  |  |     up the Idle side of the split debugger by instantiating the IdbProxy, | 
					
						
							| 
									
										
										
										
											2002-06-24 17:03:37 +00:00
										 |  |  |     debugger GUI, and debugger GUIAdapter objects and linking them together. | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  |     Register the GUIAdapter with the RPCClient to handle debugger GUI | 
					
						
							|  |  |  |     interaction requests coming from the subprocess debugger via the GUIProxy. | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     The IdbAdapter will pass execution and environment requests coming from the | 
					
						
							|  |  |  |     Idle debugger GUI to the subprocess debugger via the IdbProxy. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2002-09-05 02:31:20 +00:00
										 |  |  |     global idb_adap_oid | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  |     idb_adap_oid = rpcclt.remotecall("exec", "start_the_debugger",\ | 
					
						
							| 
									
										
										
										
											2002-06-16 03:32:24 +00:00
										 |  |  |                                    (gui_adap_oid,), {}) | 
					
						
							| 
									
										
										
										
											2003-01-25 21:33:40 +00:00
										 |  |  |     idb_proxy = IdbProxy(rpcclt, pyshell, idb_adap_oid) | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  |     gui = debugger.Debugger(pyshell, idb_proxy) | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  |     gui_adap = GUIAdapter(rpcclt, gui) | 
					
						
							|  |  |  |     rpcclt.register(gui_adap_oid, gui_adap) | 
					
						
							| 
									
										
										
										
											2002-05-26 13:36:41 +00:00
										 |  |  |     return gui | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def close_remote_debugger(rpcclt): | 
					
						
							|  |  |  |     """Shut down subprocess debugger and Idle side of debugger RPC link
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Request that the RPCServer shut down the subprocess debugger and link. | 
					
						
							|  |  |  |     Unregister the GUIAdapter, which will cause a GC on the Idle process | 
					
						
							|  |  |  |     debugger and RPC link objects.  (The second reference to the debugger GUI | 
					
						
							| 
									
										
										
										
											2016-05-28 13:22:31 -04:00
										 |  |  |     is deleted in pyshell.close_remote_debugger().) | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-12-31 16:03:23 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2002-09-05 02:31:20 +00:00
										 |  |  |     close_subprocess_debugger(rpcclt) | 
					
						
							| 
									
										
										
										
											2002-06-26 02:32:09 +00:00
										 |  |  |     rpcclt.unregister(gui_adap_oid) | 
					
						
							| 
									
										
										
										
											2002-09-05 02:31:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def close_subprocess_debugger(rpcclt): | 
					
						
							|  |  |  |     rpcclt.remotecall("exec", "stop_the_debugger", (idb_adap_oid,), {}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def restart_subprocess_debugger(rpcclt): | 
					
						
							|  |  |  |     idb_adap_oid_ret = rpcclt.remotecall("exec", "start_the_debugger",\ | 
					
						
							|  |  |  |                                          (gui_adap_oid,), {}) | 
					
						
							|  |  |  |     assert idb_adap_oid_ret == idb_adap_oid, 'Idb restarted with different oid' | 
					
						
							| 
									
										
										
										
											2018-06-15 18:20:55 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     from unittest import main | 
					
						
							| 
									
										
										
										
											2021-01-10 01:59:47 -05:00
										 |  |  |     main('idlelib.idle_test.test_debugger_r', verbosity=2, exit=False) |