summaryrefslogtreecommitdiffstats
path: root/Lib/code.py
blob: d0ff4bf2bd16fa3dc98a63e198135c0d63a49cd6 (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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
"""Utilities dealing with code objects."""

def compile_command(source, filename="<input>", symbol="single"):
    r"""Compile a command and determine whether it is incomplete.

    Arguments:

    source -- the source string; may contain \n characters
    filename -- optional filename from which source was read; default "<input>"
    symbol -- optional grammar start symbol; "single" (default) or "eval"

    Return value / exception raised:

    - Return a code object if the command is complete and valid
    - Return None if the command is incomplete
    - Raise SyntaxError if the command is a syntax error

    Approach:
    
    Compile three times: as is, with \n, and with \n\n appended.  If
    it compiles as is, it's complete.  If it compiles with one \n
    appended, we expect more.  If it doesn't compile either way, we
    compare the error we get when compiling with \n or \n\n appended.
    If the errors are the same, the code is broken.  But if the errors
    are different, we expect more.  Not intuitive; not even guaranteed
    to hold in future releases; but this matches the compiler's
    behavior in Python 1.4 and 1.5.

    """

    err = err1 = err2 = None
    code = code1 = code2 = None

    try:
        code = compile(source, filename, symbol)
    except SyntaxError, err:
        pass

    try:
        code1 = compile(source + "\n", filename, symbol)
    except SyntaxError, err1:
        pass

    try:
        code2 = compile(source + "\n\n", filename, symbol)
    except SyntaxError, err2:
        pass

    if code:
        return code
    try:
        e1 = err1.__dict__
    except AttributeError:
        e1 = err1
    try:
        e2 = err2.__dict__
    except AttributeError:
        e2 = err2
    if not code1 and e1 == e2:
        raise SyntaxError, err1


def interact(banner=None, readfunc=raw_input, local=None):
    # Due to Jeff Epler, with changes by Guido:
    """Closely emulate the interactive Python console."""
    try: import readline # Enable GNU readline if available
    except: pass
    local = local or {}
    import sys, string, traceback
    sys.ps1 = '>>> '
    sys.ps2 = '... '
    if banner:
        print banner
    else:
        print "Python Interactive Console", sys.version
        print sys.copyright
    buf = []
    while 1:
        if buf: prompt = sys.ps2
        else: prompt = sys.ps1
        try: line = readfunc(prompt)
        except KeyboardInterrupt:
            print "\nKeyboardInterrupt"
            buf = []
            continue
        except EOFError: break
        buf.append(line)
        try: x = compile_command(string.join(buf, "\n"))
        except SyntaxError:
            traceback.print_exc(0)
            buf = []
            continue
        if x == None: continue
        else:
            try: exec x in local
            except:
                exc_type, exc_value, exc_traceback = \
                        sys.exc_type, sys.exc_value, \
                        sys.exc_traceback
                l = len(traceback.extract_tb(sys.exc_traceback))
                try: 1/0
                except:
                    m = len(traceback.extract_tb(
                            sys.exc_traceback))
                traceback.print_exception(exc_type,
                        exc_value, exc_traceback, l-m)
            buf = []
                
if __name__ == '__main__':
    interact()