~azzar1/unity/add-show-desktop-key

« back to all changes in this revision

Viewing changes to console/python-console

  • Committer: William Grant
  • Date: 2012-06-28 01:52:02 UTC
  • Revision ID: me@williamgrant.id.au-20120628015202-f6ru7o367gt6nvgz
Hah

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
#!/usr/bin/python
2
 
 
3
 
# usage:
4
 
#   python-console <port> <magic>
5
 
 
6
 
import sys
7
 
import web
8
 
import md5
9
 
import codeop
10
 
import cjson
11
 
import cgi
12
 
import cStringIO
13
 
import signal
14
 
import Queue
15
 
from threading import Thread
16
 
 
17
 
class StdinFromWeb(object):
18
 
    def __init__(self, cmdQ, lineQ):
19
 
        self.cmdQ = cmdQ
20
 
        self.lineQ = lineQ
21
 
 
22
 
    def readline(self):
23
 
        # stop the clock!
24
 
        signal.alarm(0)
25
 
        self.cmdQ.put({"input":None})
26
 
        l = self.lineQ.get()
27
 
        # restart the clock:
28
 
        # Some of our 5 seconds may have elapsed, but
29
 
        # never mind.
30
 
        signal.alarm(5)
31
 
        return l
32
 
 
33
 
class PythonRunner(Thread):
34
 
    def __init__(self, cmdQ, lineQ):
35
 
        self.cmdQ = cmdQ
36
 
        self.lineQ = lineQ
37
 
        Thread.__init__(self)
38
 
 
39
 
    def run(self):
40
 
        globs = {}
41
 
        globs['__builtins__'] = globals()['__builtins__']
42
 
        locls = {}
43
 
        compiler = codeop.CommandCompiler()
44
 
        curr_cmd = ''
45
 
 
46
 
        while True:
47
 
            l = self.lineQ.get()
48
 
            if curr_cmd == '':
49
 
                curr_cmd = l
50
 
            else:
51
 
                curr_cmd = curr_cmd + '\n' + l
52
 
            try:
53
 
                cmd = compiler(curr_cmd)
54
 
                if cmd is None:
55
 
                    # The command was incomplete,
56
 
                    # so send back a None, so the
57
 
                    # client can print a '...'
58
 
                    self.cmdQ.put({"more":None})
59
 
                else:
60
 
                    # The command was complete,
61
 
                    # so evaluate it!
62
 
                    sys.stdin = StdinFromWeb(self.cmdQ, self.lineQ)
63
 
                    out = cStringIO.StringIO()
64
 
                    sys.stdout = out
65
 
                    sys.stderr = out
66
 
                    signal.alarm(5)
67
 
                    res = eval(cmd, globs, locls)
68
 
                    signal.alarm(0)
69
 
                    self.cmdQ.put({"okay":(out.getvalue(),res)})
70
 
                    curr_cmd = ''
71
 
            except Exception, exc:
72
 
                signal.alarm(0)
73
 
                self.cmdQ.put({"exc":str(exc)})
74
 
                curr_cmd = ''
75
 
 
76
 
urls = (
77
 
    '/',            'index',
78
 
    '/index.html',  'index',
79
 
    '/(.*\.js)',    'jscript',
80
 
    '/(.*\.css)',   'style',
81
 
    '/chat',        'chat')
82
 
 
83
 
# The global 'magic' is the secret that the client and server share
84
 
# which is used to create and md5 digest to authenticate requests.
85
 
# It is assigned a real value at startup.
86
 
magic = ''
87
 
 
88
 
class index:
89
 
    def GET(self):
90
 
        inp = web.input()
91
 
 
92
 
        # Authenticate
93
 
        digest = md5.new('hello' + magic).digest().encode('hex')
94
 
        if inp.digest != digest:
95
 
            web.ctx.status = '401 Unauthorized'
96
 
            return
97
 
 
98
 
        # Okay, so the authentication succeeded,
99
 
        # so all we need to do is send back the static
100
 
        # HTML for the console app.
101
 
        web.output(file("index.html", "r").read())
102
 
 
103
 
class jscript:
104
 
    def GET(self, name):
105
 
        web.output(file(name, "r").read())
106
 
 
107
 
class style:
108
 
    def GET(self, name):
109
 
        web.output(file(name, "r").read())
110
 
 
111
 
class chat:
112
 
 
113
 
    def POST(self):
114
 
        inp = web.input()
115
 
 
116
 
        # Authenticate
117
 
        digest = md5.new(inp.text + magic).digest().encode('hex')
118
 
        if inp.digest != digest:
119
 
            web.ctx.status = '401 Unauthorized'
120
 
            return
121
 
 
122
 
        # Okay, so the authentication succeeded,
123
 
        # so now we have the trivial matter of actually
124
 
        # executing the python....
125
 
        lineQ.put(inp.text)
126
 
        r = cmdQ.get()
127
 
        sys.__stderr__.write(cjson.encode(r) + "\n")
128
 
        web.output(cjson.encode(r))
129
 
 
130
 
cmdQ = Queue.Queue()
131
 
lineQ = Queue.Queue()
132
 
interpThread = PythonRunner(cmdQ, lineQ)
133
 
 
134
 
if __name__ == "__main__":
135
 
    # FIXME jail!
136
 
    magic = sys.argv[2]
137
 
    interpThread.start()
138
 
    web.run(urls, globals())