226 lines
7.1 KiB
226 lines
7.1 KiB
9 years ago
|
#!/usr/bin/env python
|
||
|
# Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
||
|
# Use of this source code is governed by a BSD-style license that can be
|
||
|
# found in the LICENSE file.
|
||
|
|
||
|
import argparse
|
||
|
import logging
|
||
|
import multiprocessing
|
||
|
import os
|
||
|
import socket
|
||
|
import sys
|
||
|
import time
|
||
|
|
||
|
if sys.version_info < (2, 7, 0):
|
||
|
sys.stderr.write("python 2.7 or later is required run this script\n")
|
||
|
sys.exit(1)
|
||
|
|
||
|
try:
|
||
|
# Python 3+
|
||
|
from http.server import HTTPServer, SimpleHTTPRequestHandler
|
||
|
from urllib.parse import urlparse, parse_qs, urlsplit
|
||
|
except ImportError:
|
||
|
# Python 2.7
|
||
|
from urlparse import urlparse, urlsplit, parse_qs
|
||
|
from BaseHTTPServer import HTTPServer
|
||
|
from SimpleHTTPServer import SimpleHTTPRequestHandler
|
||
|
|
||
|
SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
|
||
|
NACL_SDK_ROOT = os.path.dirname(SCRIPT_DIR)
|
||
|
|
||
|
# We only run from the examples directory so that not too much is exposed
|
||
|
# via this HTTP server. Everything in the directory is served, so there should
|
||
|
# never be anything potentially sensitive in the serving directory, especially
|
||
|
# if the machine might be a multi-user machine and not all users are trusted.
|
||
|
# We only serve via the loopback interface.
|
||
|
def SanityCheckDirectory(dirname):
|
||
|
abs_serve_dir = os.path.abspath(dirname)
|
||
|
|
||
|
# Verify we don't serve anywhere above NACL_SDK_ROOT.
|
||
|
if abs_serve_dir[:len(NACL_SDK_ROOT)] == NACL_SDK_ROOT:
|
||
|
return
|
||
|
logging.error('For security, httpd.py should only be run from within the')
|
||
|
logging.error('example directory tree.')
|
||
|
logging.error('Attempting to serve from %s.' % abs_serve_dir)
|
||
|
logging.error('Run with --no-dir-check to bypass this check.')
|
||
|
sys.exit(1)
|
||
|
|
||
|
|
||
|
class SuperNetHTTPServer(HTTPServer):
|
||
|
def __init__(self, *args, **kwargs):
|
||
|
HTTPServer.__init__(self, *args)
|
||
|
self.running = True
|
||
|
self.result = 0
|
||
|
|
||
|
def Shutdown(self, result=0):
|
||
|
self.running = False
|
||
|
self.result = result
|
||
|
|
||
|
|
||
|
class SuperNetHTTPRequestHandler(SimpleHTTPRequestHandler):
|
||
|
def _SendNothingAndDie(self, result=0):
|
||
|
self.send_response(200, 'OK')
|
||
|
self.send_header('Content-type', 'text/html')
|
||
|
self.send_header('Content-length', '0')
|
||
|
self.end_headers()
|
||
|
self.server.Shutdown(result)
|
||
|
|
||
|
def do_GET(self):
|
||
|
# Browsing to ?quit=1 will kill the server cleanly.
|
||
|
_, _, _, query, _ = urlsplit(self.path)
|
||
|
if query:
|
||
|
params = parse_qs(query)
|
||
|
if '1' in params.get('quit', []):
|
||
|
self._SendNothingAndDie()
|
||
|
return
|
||
|
|
||
|
return SimpleHTTPRequestHandler.do_GET(self)
|
||
|
|
||
|
|
||
|
class LocalHTTPServer(object):
|
||
|
"""Class to start a local HTTP server as a child process."""
|
||
|
|
||
|
def __init__(self, dirname, port):
|
||
|
parent_conn, child_conn = multiprocessing.Pipe()
|
||
|
self.process = multiprocessing.Process(
|
||
|
target=_HTTPServerProcess,
|
||
|
args=(child_conn, dirname, port, {}))
|
||
|
self.process.start()
|
||
|
if parent_conn.poll(10): # wait 10 seconds
|
||
|
self.port = parent_conn.recv()
|
||
|
else:
|
||
|
raise Exception('Unable to launch HTTP server.')
|
||
|
|
||
|
self.conn = parent_conn
|
||
|
|
||
|
def ServeForever(self):
|
||
|
"""Serve until the child HTTP process tells us to stop.
|
||
|
|
||
|
Returns:
|
||
|
The result from the child (as an errorcode), or 0 if the server was
|
||
|
killed not by the child (by KeyboardInterrupt for example).
|
||
|
"""
|
||
|
child_result = 0
|
||
|
try:
|
||
|
# Block on this pipe, waiting for a response from the child process.
|
||
|
child_result = self.conn.recv()
|
||
|
except KeyboardInterrupt:
|
||
|
pass
|
||
|
finally:
|
||
|
self.Shutdown()
|
||
|
return child_result
|
||
|
|
||
|
def ServeUntilSubprocessDies(self, process):
|
||
|
"""Serve until the child HTTP process tells us to stop or |subprocess| dies.
|
||
|
|
||
|
Returns:
|
||
|
The result from the child (as an errorcode), or 0 if |subprocess| died,
|
||
|
or the server was killed some other way (by KeyboardInterrupt for
|
||
|
example).
|
||
|
"""
|
||
|
child_result = 0
|
||
|
try:
|
||
|
while True:
|
||
|
if process.poll() is not None:
|
||
|
child_result = 0
|
||
|
break
|
||
|
if self.conn.poll():
|
||
|
child_result = self.conn.recv()
|
||
|
break
|
||
|
time.sleep(0)
|
||
|
except KeyboardInterrupt:
|
||
|
pass
|
||
|
finally:
|
||
|
self.Shutdown()
|
||
|
return child_result
|
||
|
|
||
|
def Shutdown(self):
|
||
|
"""Send a message to the child HTTP server process and wait for it to
|
||
|
finish."""
|
||
|
print("Shutting down server")
|
||
|
self.conn.send(False)
|
||
|
self.process.join()
|
||
|
|
||
|
def GetURL(self, rel_url):
|
||
|
"""Get the full url for a file on the local HTTP server.
|
||
|
|
||
|
Args:
|
||
|
rel_url: A URL fragment to convert to a full URL. For example,
|
||
|
GetURL('foobar.baz') -> 'http://127.0.0.1:1234/foobar.baz'
|
||
|
"""
|
||
|
return 'http://127.0.0.1:%d/%s' % (self.port, rel_url)
|
||
|
|
||
|
|
||
|
def _HTTPServerProcess(conn, dirname, port, server_kwargs):
|
||
|
"""Run a local httpserver with the given port or an ephemeral port.
|
||
|
|
||
|
This function assumes it is run as a child process using multiprocessing.
|
||
|
|
||
|
Args:
|
||
|
conn: A connection to the parent process. The child process sends
|
||
|
the local port, and waits for a message from the parent to
|
||
|
stop serving. It also sends a "result" back to the parent -- this can
|
||
|
be used to allow a client-side test to notify the server of results.
|
||
|
dirname: The directory to serve. All files are accessible through
|
||
|
http://127.0.0.1:<port>/path/to/filename.
|
||
|
port: The port to serve on. If 0, an ephemeral port will be chosen.
|
||
|
server_kwargs: A dict that will be passed as kwargs to the server.
|
||
|
"""
|
||
|
try:
|
||
|
os.chdir(dirname)
|
||
|
httpd = SuperNetHTTPServer(('', port), SuperNetHTTPRequestHandler, **server_kwargs)
|
||
|
except socket.error as e:
|
||
|
sys.stderr.write('Error creating SuperNetHTTPServer: %s\n' % e)
|
||
|
sys.exit(1)
|
||
|
|
||
|
try:
|
||
|
conn.send(httpd.server_address[1]) # the chosen port number
|
||
|
httpd.timeout = 0.5 # seconds
|
||
|
while httpd.running:
|
||
|
# Flush output for MSVS Add-In.
|
||
|
sys.stdout.flush()
|
||
|
sys.stderr.flush()
|
||
|
httpd.handle_request()
|
||
|
if conn.poll():
|
||
|
httpd.running = conn.recv()
|
||
|
except KeyboardInterrupt:
|
||
|
pass
|
||
|
finally:
|
||
|
conn.send(httpd.result)
|
||
|
conn.close()
|
||
|
|
||
|
|
||
|
def main(args):
|
||
|
parser = argparse.ArgumentParser()
|
||
|
parser.add_argument('-C', '--serve-dir',
|
||
|
help='Serve files out of this directory.',
|
||
|
default=os.path.abspath('.'))
|
||
|
parser.add_argument('-p', '--port',
|
||
|
help='Run server on this port.', default=7777)
|
||
|
parser.add_argument('--no-dir-check', '--no_dir_check',
|
||
|
help='No check to ensure serving from safe directory.',
|
||
|
dest='do_safe_check', action='store_false', default=True)
|
||
|
|
||
|
# To enable bash completion for this command first install optcomplete
|
||
|
# and then add this line to your .bashrc:
|
||
|
# complete -F _optcomplete httpd.py
|
||
|
try:
|
||
|
import optcomplete
|
||
|
optcomplete.autocomplete(parser)
|
||
|
except ImportError:
|
||
|
pass
|
||
|
|
||
|
options = parser.parse_args(args)
|
||
|
if options.do_safe_check:
|
||
|
SanityCheckDirectory(options.serve_dir)
|
||
|
|
||
|
server = LocalHTTPServer(options.serve_dir, int(options.port))
|
||
|
|
||
|
# Serve until the client tells us to stop. When it does, it will give us an
|
||
|
# errorcode.
|
||
|
print(('Serving {0} on {1}...'.format(options.serve_dir, server.GetURL(''))))
|
||
|
return server.ServeForever()
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
sys.exit(main(sys.argv[1:]))
|