server.py
442 lines
| 13.8 KiB
| text/x-python
|
PythonLexer
Eric Hopper
|
r2391 | # hgweb/server.py - The standalone hg web server. | ||
Eric Hopper
|
r2355 | # | ||
# Copyright 21 May 2005 - (c) 2005 Jake Edge <jake@edge2.net> | ||||
Thomas Arendsen Hein
|
r4635 | # Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||
Eric Hopper
|
r2355 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Eric Hopper
|
r2355 | |||
Yuya Nishihara
|
r27046 | from __future__ import absolute_import | ||
import errno | ||||
Gregory Szorc
|
r43339 | import importlib | ||
Yuya Nishihara
|
r27046 | import os | ||
import socket | ||||
import sys | ||||
import traceback | ||||
Gregory Szorc
|
r36821 | import wsgiref.validate | ||
Yuya Nishihara
|
r27046 | |||
from ..i18n import _ | ||||
Gregory Szorc
|
r43359 | from ..pycompat import ( | ||
getattr, | ||||
open, | ||||
) | ||||
Yuya Nishihara
|
r27046 | |||
from .. import ( | ||||
Augie Fackler
|
r34707 | encoding, | ||
Yuya Nishihara
|
r27046 | error, | ||
Pulkit Goyal
|
r30639 | pycompat, | ||
Yuya Nishihara
|
r27046 | util, | ||
) | ||||
Pulkit Goyal
|
r29566 | httpservermod = util.httpserver | ||
Pulkit Goyal
|
r29433 | socketserver = util.socketserver | ||
timeless
|
r28883 | urlerr = util.urlerr | ||
urlreq = util.urlreq | ||||
Augie Fackler
|
r43346 | from . import common | ||
Eric Hopper
|
r2355 | |||
def _splitURI(uri): | ||||
Mads Kiilerich
|
r17427 | """Return path and query that has been split from uri | ||
Eric Hopper
|
r2355 | |||
Just like CGI environment, the path is unquoted, the query is | ||||
not. | ||||
""" | ||||
Augie Fackler
|
r34701 | if r'?' in uri: | ||
path, query = uri.split(r'?', 1) | ||||
Eric Hopper
|
r2355 | else: | ||
Augie Fackler
|
r34701 | path, query = uri, r'' | ||
timeless
|
r28883 | return urlreq.unquote(path), query | ||
Eric Hopper
|
r2355 | |||
Augie Fackler
|
r43346 | |||
Eric Hopper
|
r2506 | class _error_logger(object): | ||
def __init__(self, handler): | ||||
self.handler = handler | ||||
Augie Fackler
|
r43346 | |||
Eric Hopper
|
r2506 | def flush(self): | ||
pass | ||||
Augie Fackler
|
r43346 | |||
Benoit Boissinot
|
r3130 | def write(self, str): | ||
Augie Fackler
|
r43347 | self.writelines(str.split(b'\n')) | ||
Augie Fackler
|
r43346 | |||
Benoit Boissinot
|
r3130 | def writelines(self, seq): | ||
Eric Hopper
|
r2506 | for msg in seq: | ||
Matt Harbison
|
r41475 | self.handler.log_error(r"HG error: %s", encoding.strfromlocal(msg)) | ||
Eric Hopper
|
r2506 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r29566 | class _httprequesthandler(httpservermod.basehttprequesthandler): | ||
Wesley J. Landaker
|
r4870 | |||
Augie Fackler
|
r43347 | url_scheme = b'http' | ||
Thomas Arendsen Hein
|
r4957 | |||
Mads Kiilerich
|
r12783 | @staticmethod | ||
Gregory Szorc
|
r29553 | def preparehttpserver(httpserver, ui): | ||
Mads Kiilerich
|
r12783 | """Prepare .socket of new HTTPServer instance""" | ||
Eric Hopper
|
r2355 | def __init__(self, *args, **kargs): | ||
Augie Fackler
|
r34513 | self.protocol_version = r'HTTP/1.1' | ||
Pulkit Goyal
|
r29566 | httpservermod.basehttprequesthandler.__init__(self, *args, **kargs) | ||
Eric Hopper
|
r2355 | |||
Patrick Mezard
|
r5549 | def _log_any(self, fp, format, *args): | ||
Augie Fackler
|
r43346 | fp.write( | ||
pycompat.sysbytes( | ||||
r"%s - - [%s] %s" | ||||
% ( | ||||
self.client_address[0], | ||||
self.log_date_time_string(), | ||||
format % args, | ||||
) | ||||
) | ||||
Augie Fackler
|
r43347 | + b'\n' | ||
Augie Fackler
|
r43346 | ) | ||
Patrick Mezard
|
r5549 | fp.flush() | ||
Eric Hopper
|
r2355 | def log_error(self, format, *args): | ||
Patrick Mezard
|
r5549 | self._log_any(self.server.errorlog, format, *args) | ||
Eric Hopper
|
r2355 | |||
def log_message(self, format, *args): | ||||
Patrick Mezard
|
r5549 | self._log_any(self.server.accesslog, format, *args) | ||
Eric Hopper
|
r2355 | |||
Augie Fackler
|
r34707 | def log_request(self, code=r'-', size=r'-'): | ||
David Soria Parra
|
r19877 | xheaders = [] | ||
Augie Fackler
|
r43347 | if util.safehasattr(self, b'headers'): | ||
Augie Fackler
|
r43346 | xheaders = [ | ||
h for h in self.headers.items() if h[0].startswith(r'x-') | ||||
] | ||||
self.log_message( | ||||
r'"%s" %s %s%s', | ||||
self.requestline, | ||||
str(code), | ||||
str(size), | ||||
r''.join([r' %s:%s' % h for h in sorted(xheaders)]), | ||||
) | ||||
Steven Brown
|
r14093 | |||
Brendan Cully
|
r4860 | def do_write(self): | ||
try: | ||||
self.do_hgweb() | ||||
Gregory Szorc
|
r25660 | except socket.error as inst: | ||
Matt Harbison
|
r40909 | if inst.errno != errno.EPIPE: | ||
Brendan Cully
|
r4860 | raise | ||
Eric Hopper
|
r2355 | def do_POST(self): | ||
try: | ||||
Brendan Cully
|
r4860 | self.do_write() | ||
Augie Fackler
|
r41620 | except Exception as e: | ||
Gregory Szorc
|
r41603 | # I/O below could raise another exception. So log the original | ||
# exception first to ensure it is recorded. | ||||
Augie Fackler
|
r43346 | if not ( | ||
isinstance(e, (OSError, socket.error)) | ||||
and e.errno == errno.ECONNRESET | ||||
): | ||||
Augie Fackler
|
r41620 | tb = r"".join(traceback.format_exception(*sys.exc_info())) | ||
# We need a native-string newline to poke in the log | ||||
# message, because we won't get a newline when using an | ||||
# r-string. This is the easy way out. | ||||
newline = chr(10) | ||||
Augie Fackler
|
r43346 | self.log_error( | ||
r"Exception happened during processing " | ||||
r"request '%s':%s%s", | ||||
self.path, | ||||
newline, | ||||
tb, | ||||
) | ||||
Eric Hopper
|
r2355 | |||
Gregory Szorc
|
r41603 | self._start_response(r"500 Internal Server Error", []) | ||
self._write(b"Internal Server Error") | ||||
self._done() | ||||
Matt Harbison
|
r37165 | def do_PUT(self): | ||
self.do_POST() | ||||
Eric Hopper
|
r2355 | def do_GET(self): | ||
self.do_POST() | ||||
def do_hgweb(self): | ||||
Augie Fackler
|
r34721 | self.sent_headers = False | ||
Michele Cella
|
r5835 | path, query = _splitURI(self.path) | ||
Eric Hopper
|
r2355 | |||
Matt Harbison
|
r37288 | # Ensure the slicing of path below is valid | ||
Augie Fackler
|
r43346 | if path != self.server.prefix and not path.startswith( | ||
self.server.prefix + b'/' | ||||
): | ||||
self._start_response(pycompat.strurl(common.statusmessage(404)), []) | ||||
Denis Laxalde
|
r43793 | if self.command == r'POST': | ||
Augie Fackler
|
r41161 | # Paranoia: tell the client we're going to close the | ||
# socket so they don't try and reuse a socket that | ||||
# might have a POST body waiting to confuse us. We do | ||||
# this by directly munging self.saved_headers because | ||||
# self._start_response ignores Connection headers. | ||||
self.saved_headers = [(r'Connection', r'Close')] | ||||
Augie Fackler
|
r38316 | self._write(b"Not Found") | ||
Matt Harbison
|
r37288 | self._done() | ||
return | ||||
Eric Hopper
|
r2355 | env = {} | ||
Augie Fackler
|
r34513 | env[r'GATEWAY_INTERFACE'] = r'CGI/1.1' | ||
env[r'REQUEST_METHOD'] = self.command | ||||
env[r'SERVER_NAME'] = self.server.server_name | ||||
env[r'SERVER_PORT'] = str(self.server.server_port) | ||||
env[r'REQUEST_URI'] = self.path | ||||
Gregory Szorc
|
r36820 | env[r'SCRIPT_NAME'] = pycompat.sysstr(self.server.prefix) | ||
Augie Fackler
|
r43346 | env[r'PATH_INFO'] = pycompat.sysstr(path[len(self.server.prefix) :]) | ||
Augie Fackler
|
r34513 | env[r'REMOTE_HOST'] = self.client_address[0] | ||
env[r'REMOTE_ADDR'] = self.client_address[0] | ||||
Gregory Szorc
|
r36821 | env[r'QUERY_STRING'] = query or r'' | ||
Eric Hopper
|
r2355 | |||
Augie Fackler
|
r34720 | if pycompat.ispy3: | ||
if self.headers.get_content_type() is None: | ||||
env[r'CONTENT_TYPE'] = self.headers.get_default_type() | ||||
else: | ||||
env[r'CONTENT_TYPE'] = self.headers.get_content_type() | ||||
Augie Fackler
|
r37609 | length = self.headers.get(r'content-length') | ||
Augie Fackler
|
r34720 | else: | ||
Augie Fackler
|
r34719 | if self.headers.typeheader is None: | ||
env[r'CONTENT_TYPE'] = self.headers.type | ||||
else: | ||||
env[r'CONTENT_TYPE'] = self.headers.typeheader | ||||
Augie Fackler
|
r37609 | length = self.headers.getheader(r'content-length') | ||
Eric Hopper
|
r2355 | if length: | ||
Augie Fackler
|
r34513 | env[r'CONTENT_LENGTH'] = length | ||
Augie Fackler
|
r43346 | for header in [ | ||
h | ||||
for h in self.headers.keys() | ||||
if h.lower() not in (r'content-type', r'content-length') | ||||
]: | ||||
Augie Fackler
|
r34513 | hkey = r'HTTP_' + header.replace(r'-', r'_').upper() | ||
hval = self.headers.get(header) | ||||
hval = hval.replace(r'\n', r'').strip() | ||||
Eric Hopper
|
r2505 | if hval: | ||
env[hkey] = hval | ||||
Augie Fackler
|
r34513 | env[r'SERVER_PROTOCOL'] = self.request_version | ||
env[r'wsgi.version'] = (1, 0) | ||||
Gregory Szorc
|
r36820 | env[r'wsgi.url_scheme'] = pycompat.sysstr(self.url_scheme) | ||
Augie Fackler
|
r43347 | if env.get(r'HTTP_EXPECT', b'').lower() == b'100-continue': | ||
Augie Fackler
|
r13570 | self.rfile = common.continuereader(self.rfile, self.wfile.write) | ||
Augie Fackler
|
r34513 | env[r'wsgi.input'] = self.rfile | ||
env[r'wsgi.errors'] = _error_logger(self) | ||||
Augie Fackler
|
r43346 | env[r'wsgi.multithread'] = isinstance( | ||
self.server, socketserver.ThreadingMixIn | ||||
) | ||||
Augie Fackler
|
r43347 | if util.safehasattr(socketserver, b'ForkingMixIn'): | ||
Augie Fackler
|
r43346 | env[r'wsgi.multiprocess'] = isinstance( | ||
self.server, socketserver.ForkingMixIn | ||||
) | ||||
Matt Harbison
|
r39866 | else: | ||
env[r'wsgi.multiprocess'] = False | ||||
Augie Fackler
|
r34513 | env[r'wsgi.run_once'] = 0 | ||
Eric Hopper
|
r2355 | |||
Gregory Szorc
|
r36821 | wsgiref.validate.check_environ(env) | ||
Eric Hopper
|
r2506 | self.saved_status = None | ||
self.saved_headers = [] | ||||
Eric Hopper
|
r2508 | self.length = None | ||
Mads Kiilerich
|
r18354 | self._chunked = None | ||
Dirkjan Ochtman
|
r6784 | for chunk in self.server.application(env, self._start_response): | ||
self._write(chunk) | ||||
Mads Kiilerich
|
r18349 | if not self.sent_headers: | ||
self.send_headers() | ||||
Mads Kiilerich
|
r18354 | self._done() | ||
Eric Hopper
|
r2506 | |||
def send_headers(self): | ||||
if not self.saved_status: | ||||
Augie Fackler
|
r43346 | raise AssertionError( | ||
Martin von Zweigbergk
|
r43387 | b"Sending headers before start_response() called" | ||
Augie Fackler
|
r43346 | ) | ||
Eric Hopper
|
r2506 | saved_status = self.saved_status.split(None, 1) | ||
saved_status[0] = int(saved_status[0]) | ||||
self.send_response(*saved_status) | ||||
Mads Kiilerich
|
r18354 | self.length = None | ||
self._chunked = False | ||||
Eric Hopper
|
r2506 | for h in self.saved_headers: | ||
self.send_header(*h) | ||||
Gregory Szorc
|
r39990 | if h[0].lower() == r'content-length': | ||
Eric Hopper
|
r2508 | self.length = int(h[1]) | ||
Augie Fackler
|
r43346 | if self.length is None and saved_status[0] != common.HTTP_NOT_MODIFIED: | ||
self._chunked = ( | ||||
not self.close_connection | ||||
and self.request_version == r'HTTP/1.1' | ||||
) | ||||
Mads Kiilerich
|
r18354 | if self._chunked: | ||
Augie Fackler
|
r34741 | self.send_header(r'Transfer-Encoding', r'chunked') | ||
Mads Kiilerich
|
r18354 | else: | ||
Augie Fackler
|
r34741 | self.send_header(r'Connection', r'close') | ||
Eric Hopper
|
r2506 | self.end_headers() | ||
self.sent_headers = True | ||||
def _start_response(self, http_status, headers, exc_info=None): | ||||
Augie Fackler
|
r38317 | assert isinstance(http_status, str) | ||
Eric Hopper
|
r2506 | code, msg = http_status.split(None, 1) | ||
code = int(code) | ||||
self.saved_status = http_status | ||||
Gregory Szorc
|
r39990 | bad_headers = (r'connection', r'transfer-encoding') | ||
Augie Fackler
|
r43346 | self.saved_headers = [ | ||
h for h in headers if h[0].lower() not in bad_headers | ||||
] | ||||
Eric Hopper
|
r2506 | return self._write | ||
def _write(self, data): | ||||
if not self.saved_status: | ||||
Augie Fackler
|
r43347 | raise AssertionError(b"data written before start_response() called") | ||
Eric Hopper
|
r2506 | elif not self.sent_headers: | ||
self.send_headers() | ||||
Eric Hopper
|
r2508 | if self.length is not None: | ||
if len(data) > self.length: | ||||
Augie Fackler
|
r43346 | raise AssertionError( | ||
Augie Fackler
|
r43347 | b"Content-length header sent, but more " | ||
b"bytes than specified are being written." | ||||
Augie Fackler
|
r43346 | ) | ||
Eric Hopper
|
r2508 | self.length = self.length - len(data) | ||
Mads Kiilerich
|
r18354 | elif self._chunked and data: | ||
Augie Fackler
|
r43347 | data = b'%x\r\n%s\r\n' % (len(data), data) | ||
Eric Hopper
|
r2506 | self.wfile.write(data) | ||
self.wfile.flush() | ||||
Eric Hopper
|
r2355 | |||
Mads Kiilerich
|
r18354 | def _done(self): | ||
if self._chunked: | ||||
Augie Fackler
|
r43347 | self.wfile.write(b'0\r\n\r\n') | ||
Mads Kiilerich
|
r18354 | self.wfile.flush() | ||
Gregory Szorc
|
r37027 | def version_string(self): | ||
if self.server.serverheader: | ||||
Yuya Nishihara
|
r38613 | return encoding.strfromlocal(self.server.serverheader) | ||
Gregory Szorc
|
r37027 | return httpservermod.basehttprequesthandler.version_string(self) | ||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r12784 | class _httprequesthandlerssl(_httprequesthandler): | ||
timeless@mozdev.org
|
r26202 | """HTTPS handler based on Python's ssl module""" | ||
Mads Kiilerich
|
r12784 | |||
Augie Fackler
|
r43347 | url_scheme = b'https' | ||
Mads Kiilerich
|
r12784 | |||
@staticmethod | ||||
Gregory Szorc
|
r29553 | def preparehttpserver(httpserver, ui): | ||
Mads Kiilerich
|
r12784 | try: | ||
Gregory Szorc
|
r29555 | from .. import sslutil | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r29555 | sslutil.modernssl | ||
Mads Kiilerich
|
r12784 | except ImportError: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"SSL support is unavailable")) | ||
Gregory Szorc
|
r29553 | |||
Augie Fackler
|
r43347 | certfile = ui.config(b'web', b'certificate') | ||
Gregory Szorc
|
r29555 | |||
# These config options are currently only meant for testing. Use | ||||
# at your own risk. | ||||
Augie Fackler
|
r43347 | cafile = ui.config(b'devel', b'servercafile') | ||
reqcert = ui.configbool(b'devel', b'serverrequirecert') | ||||
Gregory Szorc
|
r29555 | |||
Augie Fackler
|
r43346 | httpserver.socket = sslutil.wrapserversocket( | ||
httpserver.socket, | ||||
ui, | ||||
certfile=certfile, | ||||
cafile=cafile, | ||||
requireclientcert=reqcert, | ||||
) | ||||
Mads Kiilerich
|
r12784 | |||
def setup(self): | ||||
self.connection = self.request | ||||
Augie Fackler
|
r36797 | self.rfile = self.request.makefile(r"rb", self.rbufsize) | ||
self.wfile = self.request.makefile(r"wb", self.wbufsize) | ||||
Mads Kiilerich
|
r12784 | |||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r10639 | try: | ||
Yuya Nishihara
|
r27046 | import threading | ||
Augie Fackler
|
r43346 | |||
threading.activeCount() # silence pyflakes and bypass demandimport | ||||
Pulkit Goyal
|
r29433 | _mixin = socketserver.ThreadingMixIn | ||
Dirkjan Ochtman
|
r10639 | except ImportError: | ||
Augie Fackler
|
r43347 | if util.safehasattr(os, b"fork"): | ||
Pulkit Goyal
|
r29433 | _mixin = socketserver.ForkingMixIn | ||
Dirkjan Ochtman
|
r10639 | else: | ||
Augie Fackler
|
r43346 | |||
Thomas Arendsen Hein
|
r14764 | class _mixin(object): | ||
Dirkjan Ochtman
|
r10639 | pass | ||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r10643 | def openlog(opt, default): | ||
Augie Fackler
|
r43347 | if opt and opt != b'-': | ||
return open(opt, b'ab') | ||||
Dirkjan Ochtman
|
r10643 | return default | ||
Augie Fackler
|
r43346 | |||
Martijn Pieters
|
r30082 | class MercurialHTTPServer(_mixin, httpservermod.httpserver, object): | ||
Dirkjan Ochtman
|
r10643 | |||
# SO_REUSEADDR has broken semantics on windows | ||||
Jun Wu
|
r34646 | if pycompat.iswindows: | ||
Dirkjan Ochtman
|
r10643 | allow_reuse_address = 0 | ||
def __init__(self, ui, app, addr, handler, **kwargs): | ||||
Pulkit Goyal
|
r29566 | httpservermod.httpserver.__init__(self, addr, handler, **kwargs) | ||
Dirkjan Ochtman
|
r10643 | self.daemon_threads = True | ||
self.application = app | ||||
Eric Hopper
|
r2355 | |||
Gregory Szorc
|
r29553 | handler.preparehttpserver(self, ui) | ||
Dirkjan Ochtman
|
r10643 | |||
Augie Fackler
|
r43347 | prefix = ui.config(b'web', b'prefix') | ||
Dirkjan Ochtman
|
r10643 | if prefix: | ||
Augie Fackler
|
r43347 | prefix = b'/' + prefix.strip(b'/') | ||
Dirkjan Ochtman
|
r10643 | self.prefix = prefix | ||
Augie Fackler
|
r43347 | alog = openlog(ui.config(b'web', b'accesslog'), ui.fout) | ||
elog = openlog(ui.config(b'web', b'errorlog'), ui.ferr) | ||||
Dirkjan Ochtman
|
r10643 | self.accesslog = alog | ||
self.errorlog = elog | ||||
self.addr, self.port = self.socket.getsockname()[0:2] | ||||
self.fqaddr = socket.getfqdn(addr[0]) | ||||
Augie Fackler
|
r43347 | self.serverheader = ui.config(b'web', b'server-header') | ||
Gregory Szorc
|
r37027 | |||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r10643 | class IPv6HTTPServer(MercurialHTTPServer): | ||
address_family = getattr(socket, 'AF_INET6', None) | ||||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r10643 | def __init__(self, *args, **kwargs): | ||
if self.address_family is None: | ||||
Augie Fackler
|
r43347 | raise error.RepoError(_(b'IPv6 is not available on this system')) | ||
Dirkjan Ochtman
|
r10643 | super(IPv6HTTPServer, self).__init__(*args, **kwargs) | ||
Augie Fackler
|
r43346 | |||
Dirkjan Ochtman
|
r10644 | def create_server(ui, app): | ||
Eric Hopper
|
r2355 | |||
Augie Fackler
|
r43347 | if ui.config(b'web', b'certificate'): | ||
Siddharth Agarwal
|
r26848 | handler = _httprequesthandlerssl | ||
Brendan Cully
|
r4860 | else: | ||
Mads Kiilerich
|
r12783 | handler = _httprequesthandler | ||
Brendan Cully
|
r4860 | |||
Augie Fackler
|
r43347 | if ui.configbool(b'web', b'ipv6'): | ||
Dirkjan Ochtman
|
r10641 | cls = IPv6HTTPServer | ||
else: | ||||
cls = MercurialHTTPServer | ||||
Dirkjan Ochtman
|
r8224 | # ugly hack due to python issue5853 (for threaded use) | ||
Matt Mackall
|
r20357 | try: | ||
import mimetypes | ||||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r20357 | mimetypes.init() | ||
except UnicodeDecodeError: | ||||
# Python 2.x's mimetypes module attempts to decode strings | ||||
# from Windows' ANSI APIs as ascii (fail), then re-encode them | ||||
# as ascii (clown fail), because the default Python Unicode | ||||
# codec is hardcoded as ascii. | ||||
Augie Fackler
|
r43346 | sys.argv # unwrap demand-loader so that reload() works | ||
Gregory Szorc
|
r43339 | # resurrect sys.setdefaultencoding() | ||
try: | ||||
importlib.reload(sys) | ||||
except AttributeError: | ||||
reload(sys) | ||||
Matt Mackall
|
r20357 | oldenc = sys.getdefaultencoding() | ||
Augie Fackler
|
r43347 | sys.setdefaultencoding(b"latin1") # or any full 8-bit encoding | ||
Matt Mackall
|
r20357 | mimetypes.init() | ||
sys.setdefaultencoding(oldenc) | ||||
Dirkjan Ochtman
|
r8224 | |||
Augie Fackler
|
r43347 | address = ui.config(b'web', b'address') | ||
port = util.getport(ui.config(b'web', b'port')) | ||||
Matt Mackall
|
r3628 | try: | ||
Dirkjan Ochtman
|
r10644 | return cls(ui, app, (address, port), handler) | ||
Gregory Szorc
|
r25660 | except socket.error as inst: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b"cannot start server at '%s:%d': %s") | ||
Augie Fackler
|
r43346 | % (address, port, encoding.strtolocal(inst.args[1])) | ||
) | ||||