tinyproxy.py
218 lines
| 6.5 KiB
| text/x-python
|
PythonLexer
/ tests / tinyproxy.py
r48294 | #!/usr/bin/env python | |||
Vadim Gelfer
|
r2337 | |||
Gregory Szorc
|
r27302 | |||
Vadim Gelfer
|
r2337 | __doc__ = """Tiny HTTP Proxy. | ||
This module implements GET, HEAD, POST, PUT and DELETE methods | ||||
on BaseHTTPServer, and behaves as an HTTP proxy. The CONNECT | ||||
method is also implemented experimentally, but has not been | ||||
tested yet. | ||||
Any help will be greatly appreciated. SUZUKI Hisao | ||||
""" | ||||
__version__ = "0.2.1" | ||||
Pulkit Goyal
|
r29565 | import optparse | ||
Gregory Szorc
|
r27302 | import os | ||
import select | ||||
import socket | ||||
Yuya Nishihara
|
r28778 | import sys | ||
Pulkit Goyal
|
r29431 | |||
Gregory Szorc
|
r41858 | from mercurial import ( | ||
pycompat, | ||||
util, | ||||
) | ||||
Pulkit Goyal
|
r29431 | |||
Pulkit Goyal
|
r29566 | httpserver = util.httpserver | ||
Pulkit Goyal
|
r29433 | socketserver = util.socketserver | ||
Gregory Szorc
|
r31571 | urlreq = util.urlreq | ||
Vadim Gelfer
|
r2337 | |||
Jun Wu
|
r31005 | if os.environ.get('HGIPV6', '0') == '1': | ||
family = socket.AF_INET6 | ||||
else: | ||||
family = socket.AF_INET | ||||
Augie Fackler
|
r43346 | |||
class ProxyHandler(httpserver.basehttprequesthandler): | ||||
Pulkit Goyal
|
r29566 | __base = httpserver.basehttprequesthandler | ||
Vadim Gelfer
|
r2337 | __base_handle = __base.handle | ||
server_version = "TinyHTTPProxy/" + __version__ | ||||
Augie Fackler
|
r43346 | rbufsize = 0 # self.rfile Be unbuffered | ||
Vadim Gelfer
|
r2337 | |||
def handle(self): | ||||
timeless
|
r27637 | (ip, port) = self.client_address | ||
Augie Fackler
|
r14971 | allowed = getattr(self, 'allowed_clients', None) | ||
if allowed is not None and ip not in allowed: | ||||
Vadim Gelfer
|
r2337 | self.raw_requestline = self.rfile.readline() | ||
Matt Mackall
|
r10282 | if self.parse_request(): | ||
self.send_error(403) | ||||
Vadim Gelfer
|
r2337 | else: | ||
self.__base_handle() | ||||
Steven Brown
|
r14093 | def log_request(self, code='-', size='-'): | ||
xheaders = [h for h in self.headers.items() if h[0].startswith('x-')] | ||||
Augie Fackler
|
r43346 | self.log_message( | ||
'"%s" %s %s%s', | ||||
self.requestline, | ||||
str(code), | ||||
str(size), | ||||
''.join([' %s:%s' % h for h in sorted(xheaders)]), | ||||
) | ||||
Matt Harbison
|
r32906 | # Flush for Windows, so output isn't lost on TerminateProcess() | ||
Matt Harbison
|
r32916 | sys.stdout.flush() | ||
Matt Harbison
|
r32906 | sys.stderr.flush() | ||
Steven Brown
|
r14093 | |||
Vadim Gelfer
|
r2337 | def _connect_to(self, netloc, soc): | ||
i = netloc.find(':') | ||||
if i >= 0: | ||||
Augie Fackler
|
r43346 | host_port = netloc[:i], int(netloc[i + 1 :]) | ||
Vadim Gelfer
|
r2337 | else: | ||
host_port = netloc, 80 | ||||
Pulkit Goyal
|
r28646 | print("\t" "connect to %s:%d" % host_port) | ||
Augie Fackler
|
r43346 | try: | ||
soc.connect(host_port) | ||||
Manuel Jacob
|
r50169 | except socket.error as e: | ||
self.send_error(404, e.strerror) | ||||
Vadim Gelfer
|
r2337 | return 0 | ||
return 1 | ||||
def do_CONNECT(self): | ||||
Jun Wu
|
r31005 | soc = socket.socket(family, socket.SOCK_STREAM) | ||
Vadim Gelfer
|
r2337 | try: | ||
if self._connect_to(self.path, soc): | ||||
self.log_request(200) | ||||
Augie Fackler
|
r43346 | self.wfile.write( | ||
pycompat.bytestr(self.protocol_version) | ||||
+ b" 200 Connection established\r\n" | ||||
) | ||||
self.wfile.write( | ||||
b"Proxy-agent: %s\r\n" | ||||
% pycompat.bytestr(self.version_string()) | ||||
) | ||||
Gregory Szorc
|
r41858 | self.wfile.write(b"\r\n") | ||
Vadim Gelfer
|
r2337 | self._read_write(soc, 300) | ||
finally: | ||||
Pulkit Goyal
|
r28646 | print("\t" "bye") | ||
Vadim Gelfer
|
r2337 | soc.close() | ||
self.connection.close() | ||||
def do_GET(self): | ||||
Gregory Szorc
|
r31571 | (scm, netloc, path, params, query, fragment) = urlreq.urlparse( | ||
Augie Fackler
|
r43346 | self.path, 'http' | ||
) | ||||
Vadim Gelfer
|
r2337 | if scm != 'http' or fragment or not netloc: | ||
self.send_error(400, "bad url %s" % self.path) | ||||
return | ||||
Jun Wu
|
r31005 | soc = socket.socket(family, socket.SOCK_STREAM) | ||
Vadim Gelfer
|
r2337 | try: | ||
if self._connect_to(netloc, soc): | ||||
self.log_request() | ||||
Gregory Szorc
|
r41858 | url = urlreq.urlunparse(('', '', path, params, query, '')) | ||
Augie Fackler
|
r43346 | soc.send( | ||
b"%s %s %s\r\n" | ||||
% ( | ||||
pycompat.bytestr(self.command), | ||||
pycompat.bytestr(url), | ||||
pycompat.bytestr(self.request_version), | ||||
) | ||||
) | ||||
Vadim Gelfer
|
r2337 | self.headers['Connection'] = 'close' | ||
del self.headers['Proxy-Connection'] | ||||
Gregory Szorc
|
r41858 | for key, val in self.headers.items(): | ||
Augie Fackler
|
r43346 | soc.send( | ||
b"%s: %s\r\n" | ||||
% (pycompat.bytestr(key), pycompat.bytestr(val)) | ||||
) | ||||
Gregory Szorc
|
r41858 | soc.send(b"\r\n") | ||
Vadim Gelfer
|
r2337 | self._read_write(soc) | ||
finally: | ||||
Pulkit Goyal
|
r28646 | print("\t" "bye") | ||
Vadim Gelfer
|
r2337 | soc.close() | ||
self.connection.close() | ||||
def _read_write(self, soc, max_idling=20): | ||||
iw = [self.connection, soc] | ||||
ow = [] | ||||
count = 0 | ||||
Martin Geisler
|
r14494 | while True: | ||
Vadim Gelfer
|
r2337 | count += 1 | ||
(ins, _, exs) = select.select(iw, ow, iw, 3) | ||||
Matt Mackall
|
r10282 | if exs: | ||
break | ||||
Vadim Gelfer
|
r2337 | if ins: | ||
for i in ins: | ||||
if i is soc: | ||||
out = self.connection | ||||
else: | ||||
out = soc | ||||
Mads Kiilerich
|
r18519 | try: | ||
data = i.recv(8192) | ||||
except socket.error: | ||||
break | ||||
Vadim Gelfer
|
r2337 | if data: | ||
out.send(data) | ||||
count = 0 | ||||
else: | ||||
Pulkit Goyal
|
r28646 | print("\t" "idle", count) | ||
Matt Mackall
|
r10282 | if count == max_idling: | ||
break | ||||
Vadim Gelfer
|
r2337 | |||
do_HEAD = do_GET | ||||
do_POST = do_GET | ||||
Augie Fackler
|
r43346 | do_PUT = do_GET | ||
Matt Mackall
|
r10282 | do_DELETE = do_GET | ||
Vadim Gelfer
|
r2337 | |||
Augie Fackler
|
r43346 | |||
class ThreadingHTTPServer(socketserver.ThreadingMixIn, httpserver.httpserver): | ||||
Matt Mackall
|
r16300 | def __init__(self, *args, **kwargs): | ||
Pulkit Goyal
|
r29566 | httpserver.httpserver.__init__(self, *args, **kwargs) | ||
Matt Mackall
|
r16300 | a = open("proxy.pid", "w") | ||
a.write(str(os.getpid()) + "\n") | ||||
a.close() | ||||
Vadim Gelfer
|
r2337 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r29565 | def runserver(port=8000, bind=""): | ||
server_address = (bind, port) | ||||
ProxyHandler.protocol_version = "HTTP/1.0" | ||||
httpd = ThreadingHTTPServer(server_address, ProxyHandler) | ||||
sa = httpd.socket.getsockname() | ||||
print("Serving HTTP on", sa[0], "port", sa[1], "...") | ||||
try: | ||||
httpd.serve_forever() | ||||
except KeyboardInterrupt: | ||||
print("\nKeyboard interrupt received, exiting.") | ||||
httpd.server_close() | ||||
sys.exit(0) | ||||
Augie Fackler
|
r43346 | |||
Vadim Gelfer
|
r2337 | if __name__ == '__main__': | ||
Yuya Nishihara
|
r28778 | argv = sys.argv | ||
Vadim Gelfer
|
r2337 | if argv[1:] and argv[1] in ('-h', '--help'): | ||
Pulkit Goyal
|
r28646 | print(argv[0], "[port [allowed_client_name ...]]") | ||
Vadim Gelfer
|
r2337 | else: | ||
if argv[2:]: | ||||
allowed = [] | ||||
for name in argv[2:]: | ||||
client = socket.gethostbyname(name) | ||||
allowed.append(client) | ||||
Pulkit Goyal
|
r28646 | print("Accept: %s (%s)" % (client, name)) | ||
Vadim Gelfer
|
r2337 | ProxyHandler.allowed_clients = allowed | ||
del argv[2:] | ||||
else: | ||||
Pulkit Goyal
|
r28646 | print("Any clients will be served...") | ||
Pulkit Goyal
|
r29565 | |||
parser = optparse.OptionParser() | ||||
Augie Fackler
|
r43346 | parser.add_option( | ||
'-b', | ||||
'--bind', | ||||
metavar='ADDRESS', | ||||
help='Specify alternate bind address ' '[default: all interfaces]', | ||||
default='', | ||||
) | ||||
Pulkit Goyal
|
r29565 | (options, args) = parser.parse_args() | ||
port = 8000 | ||||
if len(args) == 1: | ||||
port = int(args[0]) | ||||
runserver(port, options.bind) | ||||