Guido van Rossum | 5c8b991 | 1997-07-19 21:00:47 +0000 | [diff] [blame] | 1 | #! /usr/bin/env python |
| 2 | |
| 3 | """A multi-threaded telnet-like server that gives a Python prompt. |
| 4 | |
| 5 | This is really a prototype for the same thing in C. |
| 6 | |
| 7 | Usage: pysvr.py [port] |
| 8 | |
Guido van Rossum | eca991d | 1997-07-19 21:13:53 +0000 | [diff] [blame^] | 9 | For security reasons, it only accepts requests from the current host. |
| 10 | This can still be insecure, but restricts violations from people who |
| 11 | can log in on your machine. Use with caution! |
| 12 | |
Guido van Rossum | 5c8b991 | 1997-07-19 21:00:47 +0000 | [diff] [blame] | 13 | """ |
| 14 | |
| 15 | import sys, os, string, getopt, thread, socket, traceback |
| 16 | |
Guido van Rossum | eca991d | 1997-07-19 21:13:53 +0000 | [diff] [blame^] | 17 | PORT = 4000 # Default port |
Guido van Rossum | 5c8b991 | 1997-07-19 21:00:47 +0000 | [diff] [blame] | 18 | |
| 19 | def main(): |
| 20 | try: |
| 21 | opts, args = getopt.getopt(sys.argv[1:], "") |
| 22 | if len(args) > 1: |
| 23 | raise getopt.error, "Too many arguments." |
| 24 | except getopt.error, msg: |
| 25 | usage(msg) |
| 26 | for o, a in opts: |
| 27 | pass |
| 28 | if args: |
| 29 | try: |
| 30 | port = string.atoi(args[0]) |
| 31 | except ValueError, msg: |
| 32 | usage(msg) |
| 33 | else: |
| 34 | port = PORT |
| 35 | main_thread(port) |
| 36 | |
| 37 | def usage(msg=None): |
| 38 | sys.stdout = sys.stderr |
| 39 | if msg: |
| 40 | print msg |
| 41 | print "\n", __doc__, |
| 42 | sys.exit(2) |
| 43 | |
| 44 | def main_thread(port): |
| 45 | sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) |
| 46 | sock.bind(("", port)) |
| 47 | sock.listen(5) |
Guido van Rossum | eca991d | 1997-07-19 21:13:53 +0000 | [diff] [blame^] | 48 | print "Listening on port", port, "..." |
Guido van Rossum | 5c8b991 | 1997-07-19 21:00:47 +0000 | [diff] [blame] | 49 | while 1: |
| 50 | (conn, addr) = sock.accept() |
| 51 | thread.start_new_thread(service_thread, (conn, addr)) |
| 52 | del conn, addr |
| 53 | |
| 54 | def service_thread(conn, addr): |
| 55 | (caddr, cport) = addr |
Guido van Rossum | eca991d | 1997-07-19 21:13:53 +0000 | [diff] [blame^] | 56 | if caddr != socket.gethostbyname(socket.gethostname()): |
| 57 | print "Connection from", caddr, "not accepted." |
Guido van Rossum | 5c8b991 | 1997-07-19 21:00:47 +0000 | [diff] [blame] | 58 | return |
| 59 | print "Thread %s has connection from %s.\n" % (str(thread.get_ident()), |
Guido van Rossum | eca991d | 1997-07-19 21:13:53 +0000 | [diff] [blame^] | 60 | caddr), |
Guido van Rossum | 5c8b991 | 1997-07-19 21:00:47 +0000 | [diff] [blame] | 61 | stdin = conn.makefile("r") |
| 62 | stdout = conn.makefile("w", 0) |
| 63 | run_interpreter(stdin, stdout) |
| 64 | print "Thread %s is done.\n" % str(thread.get_ident()), |
| 65 | |
| 66 | def run_interpreter(stdin, stdout): |
| 67 | globals = {} |
| 68 | try: |
| 69 | str(sys.ps1) |
| 70 | except: |
| 71 | sys.ps1 = ">>> " |
| 72 | source = "" |
| 73 | while 1: |
| 74 | stdout.write(sys.ps1) |
| 75 | line = stdin.readline() |
| 76 | if line[:2] == '\377\354': |
| 77 | line = "" |
| 78 | if not line and not source: |
| 79 | break |
| 80 | if line[-2:] == '\r\n': |
| 81 | line = line[:-2] + '\n' |
| 82 | source = source + line |
| 83 | try: |
| 84 | code = compile_command(source) |
| 85 | except SyntaxError, err: |
| 86 | source = "" |
| 87 | traceback.print_exception(SyntaxError, err, None, file=stdout) |
| 88 | continue |
| 89 | if not code: |
| 90 | continue |
| 91 | source = "" |
| 92 | try: |
| 93 | run_command(code, stdin, stdout, globals) |
| 94 | except SystemExit, how: |
| 95 | if how: |
| 96 | try: |
| 97 | how = str(how) |
| 98 | except: |
| 99 | how = "" |
| 100 | stdout.write("Exit %s\n" % how) |
| 101 | break |
| 102 | stdout.write("\nGoodbye.\n") |
| 103 | |
| 104 | def run_command(code, stdin, stdout, globals): |
| 105 | save = sys.stdin, sys.stdout, sys.stderr |
| 106 | try: |
| 107 | sys.stdout = sys.stderr = stdout |
| 108 | sys.stdin = stdin |
| 109 | try: |
| 110 | exec code in globals |
| 111 | except SystemExit, how: |
| 112 | raise SystemExit, how, sys.exc_info()[2] |
| 113 | except: |
| 114 | type, value, tb = sys.exc_info() |
| 115 | if tb: tb = tb.tb_next |
| 116 | traceback.print_exception(type, value, tb) |
| 117 | del tb |
| 118 | finally: |
| 119 | sys.stdin, sys.stdout, sys.stderr = save |
| 120 | |
| 121 | from code import compile_command |
| 122 | |
| 123 | main() |