summaryrefslogtreecommitdiffstats
path: root/Lib/idlelib/run.py
blob: 9ede2ff48eca51e1b303346f3fc7a1fe6abe5d57 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
import sys
import time
import socket
import __main__
import rpc

def main():
    """Start the Python execution server in a subprocess

    In Idle, RPCServer is instantiated with handlerclass MyHandler, which
    inherits register/unregister methods from RPCHandler via the mix-in class
    SocketIO.

    When the RPCServer is instantiated, the TCPServer initialization creates an
    instance of run.MyHandler and calls its handle() method.  handle()
    instantiates a run.Executive, passing it a reference to the MyHandler
    object.  That reference is saved as an attribute of the Executive instance.
    The Executive methods have access to the reference and can pass it on to
    entities that they command (e.g. RemoteDebugger.Debugger.start_debugger()).
    The latter, in turn, can call MyHandler(SocketIO) register/unregister
    methods via the reference to register and unregister themselves.

    """
    port = 8833
    if sys.argv[1:]:
        port = int(sys.argv[1])
    sys.argv[:] = [""]
    addr = ("localhost", port)
    for i in range(6):
        time.sleep(i)
        try:
            svr = rpc.RPCServer(addr, MyHandler)
            break
        except socket.error, err:
            if i < 3:
                print>>sys.__stderr__, ".. ",
            else:
                print>>sys.__stderr__,"\nPython subprocess socket error: "\
                                              + err[1] + ", retrying...."
    else:
        print>>sys.__stderr__, "\nConnection to Idle failed, exiting."
        sys.exit()
    svr.handle_request() # A single request only

class MyHandler(rpc.RPCHandler):

    def handle(self):
        executive = Executive(self)
        self.register("exec", executive)
        sys.stdin = self.get_remote_proxy("stdin")
        sys.stdout = self.get_remote_proxy("stdout")
        sys.stderr = self.get_remote_proxy("stderr")
        rpc.RPCHandler.handle(self)

class Executive:

    def __init__(self, rpchandler):
        self.rpchandler = rpchandler
        self.base_env_keys = __main__.__dict__.keys()

    def runcode(self, code):
       exec code in __main__.__dict__
        
    def clear_the_environment(self):
        global __main__
        env = __main__.__dict__
        for key in env.keys():
            if key not in self.base_env_keys:
                del env[key]
        env['__doc__'] = None

    def start_the_debugger(self, gui_adap_oid):
        import RemoteDebugger
        return RemoteDebugger.start_debugger(self.rpchandler, gui_adap_oid)

    def stop_the_debugger(self, idb_adap_oid):
        "Unregister the Idb Adapter.  Link objects and Idb then subject to GC"
        self.rpchandler.unregister(idb_adap_oid)

    def stackviewer(self, flist_oid=None):
        if not hasattr(sys, "last_traceback"):
            return None
        flist = None
        if flist_oid is not None:
            flist = self.rpchandler.get_remote_proxy(flist_oid)
        import RemoteObjectBrowser
        import StackViewer
        tb = sys.last_traceback
        while tb and tb.tb_frame.f_globals["__name__"] in ["rpc", "run"]:
            tb = tb.tb_next
        item = StackViewer.StackTreeItem(flist, tb)
        return RemoteObjectBrowser.remote_object_tree_item(item)