263 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
#! /usr/bin/env python
 | 
						|
 | 
						|
# Shamefully copied from this gist :
 | 
						|
# https://gist.github.com/pankajp/280596a5dabaeeceaaaa/
 | 
						|
 | 
						|
# Standard library imports.
 | 
						|
import sys
 | 
						|
 | 
						|
if sys.version_info[0] < 3:
 | 
						|
    from SocketServer import ThreadingMixIn
 | 
						|
    import BaseHTTPServer
 | 
						|
    from SimpleHTTPServer import SimpleHTTPRequestHandler
 | 
						|
    from urllib import quote, unquote
 | 
						|
    from cStringIO import StringIO
 | 
						|
else:
 | 
						|
    import http.server as BaseHTTPServer
 | 
						|
    from http.server import SimpleHTTPRequestHandler
 | 
						|
    from socketserver import ThreadingMixIn
 | 
						|
    from urllib.parse import quote, unquote
 | 
						|
    from io import BytesIO as cStringIO
 | 
						|
 | 
						|
import os
 | 
						|
from os.path import (join, exists, abspath, split, splitdrive, isdir)
 | 
						|
from os import makedirs, unlink, getcwd, chdir, curdir, pardir, rename, fstat
 | 
						|
from shutil import copyfileobj, copytree
 | 
						|
import glob
 | 
						|
from zipfile import ZipFile
 | 
						|
from posixpath import normpath
 | 
						|
import re
 | 
						|
import cgi
 | 
						|
import threading
 | 
						|
import socket
 | 
						|
import errno
 | 
						|
 | 
						|
DATA_DIR = getcwd()
 | 
						|
 | 
						|
 | 
						|
class ThreadingHTTPServer(ThreadingMixIn, BaseHTTPServer.HTTPServer):
 | 
						|
    pass
 | 
						|
 | 
						|
 | 
						|
class RequestHandler(SimpleHTTPRequestHandler):
 | 
						|
    """ Handler to handle POST requests for actions.
 | 
						|
    """
 | 
						|
 | 
						|
    serve_path = DATA_DIR
 | 
						|
 | 
						|
    def do_GET(self):
 | 
						|
        """ Overridden to handle HTTP Range requests. """
 | 
						|
        self.range_from, self.range_to = self._get_range_header()
 | 
						|
        if self.range_from is None:
 | 
						|
            # nothing to do here
 | 
						|
            return SimpleHTTPRequestHandler.do_GET(self)
 | 
						|
        f = self.send_range_head()
 | 
						|
        if f:
 | 
						|
            self.copy_file_range(f, self.wfile)
 | 
						|
            f.close()
 | 
						|
 | 
						|
    def copy_file_range(self, in_file, out_file):
 | 
						|
        """ Copy only the range in self.range_from/to. """
 | 
						|
        in_file.seek(self.range_from)
 | 
						|
        # Add 1 because the range is inclusive
 | 
						|
        left_to_copy = 1 + self.range_to - self.range_from
 | 
						|
        buf_length = 64 * 1024
 | 
						|
        bytes_copied = 0
 | 
						|
        while bytes_copied < left_to_copy:
 | 
						|
            read_buf = in_file.read(min(buf_length, left_to_copy - bytes_copied))
 | 
						|
            if len(read_buf) == 0:
 | 
						|
                break
 | 
						|
            out_file.write(read_buf)
 | 
						|
            bytes_copied += len(read_buf)
 | 
						|
        return bytes_copied
 | 
						|
 | 
						|
    def send_range_head(self):
 | 
						|
        """Common code for GET and HEAD commands.
 | 
						|
 | 
						|
        This sends the response code and MIME headers.
 | 
						|
 | 
						|
        Return value is either a file object (which has to be copied
 | 
						|
        to the outputfile by the caller unless the command was HEAD,
 | 
						|
        and must be closed by the caller under all circumstances), or
 | 
						|
        None, in which case the caller has nothing further to do.
 | 
						|
 | 
						|
        """
 | 
						|
        path = self.translate_path(self.path)
 | 
						|
        f = None
 | 
						|
        if isdir(path):
 | 
						|
            if not self.path.endswith('/'):
 | 
						|
                # redirect browser - doing basically what apache does
 | 
						|
                self.send_response(301)
 | 
						|
                self.send_header("Location", self.path + "/")
 | 
						|
                self.end_headers()
 | 
						|
                return None
 | 
						|
            for index in "index.html", "index.htm":
 | 
						|
                index = join(path, index)
 | 
						|
                if exists(index):
 | 
						|
                    path = index
 | 
						|
                    break
 | 
						|
            else:
 | 
						|
                return self.list_directory(path)
 | 
						|
 | 
						|
        if not exists(path) and path.endswith('/data'):
 | 
						|
            # FIXME: Handle grits-like query with /data appended to path
 | 
						|
            # stupid grits
 | 
						|
            if exists(path[:-5]):
 | 
						|
                path = path[:-5]
 | 
						|
 | 
						|
        ctype = self.guess_type(path)
 | 
						|
        try:
 | 
						|
            # Always read in binary mode. Opening files in text mode may cause
 | 
						|
            # newline translations, making the actual size of the content
 | 
						|
            # transmitted *less* than the content-length!
 | 
						|
            f = open(path, 'rb')
 | 
						|
        except IOError:
 | 
						|
            self.send_error(404, "File not found")
 | 
						|
            return None
 | 
						|
 | 
						|
        if self.range_from is None:
 | 
						|
            self.send_response(200)
 | 
						|
        else:
 | 
						|
            self.send_response(206)
 | 
						|
 | 
						|
        self.send_header("Content-type", ctype)
 | 
						|
        fs = fstat(f.fileno())
 | 
						|
        file_size = fs.st_size
 | 
						|
        if self.range_from is not None:
 | 
						|
            if self.range_to is None or self.range_to >= file_size:
 | 
						|
                self.range_to = file_size - 1
 | 
						|
            self.send_header("Content-Range",
 | 
						|
                             "bytes %d-%d/%d" % (self.range_from,
 | 
						|
                                                 self.range_to,
 | 
						|
                                                 file_size))
 | 
						|
            # Add 1 because ranges are inclusive
 | 
						|
            self.send_header("Content-Length",
 | 
						|
                             (1 + self.range_to - self.range_from))
 | 
						|
        else:
 | 
						|
            self.send_header("Content-Length", str(file_size))
 | 
						|
        self.send_header("Last-Modified", self.date_time_string(fs.st_mtime))
 | 
						|
        self.end_headers()
 | 
						|
        return f
 | 
						|
 | 
						|
    def list_directory(self, path):
 | 
						|
        """Helper to produce a directory listing (absent index.html).
 | 
						|
 | 
						|
        Return value is either a file object, or None (indicating an
 | 
						|
        error).  In either case, the headers are sent, making the
 | 
						|
        interface the same as for send_head().
 | 
						|
 | 
						|
        """
 | 
						|
        try:
 | 
						|
            list = os.listdir(path)
 | 
						|
        except os.error:
 | 
						|
            self.send_error(404, "No permission to list directory")
 | 
						|
            return None
 | 
						|
        list.sort(key=lambda a: a.lower())
 | 
						|
        f = StringIO()
 | 
						|
        displaypath = cgi.escape(unquote(self.path))
 | 
						|
        f.write('<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">')
 | 
						|
        f.write("<html>\n<title>Directory listing for %s</title>\n" % displaypath)
 | 
						|
        f.write("<body>\n<h2>Directory listing for %s</h2>\n" % displaypath)
 | 
						|
        f.write("<hr>\n<ul>\n")
 | 
						|
        for name in list:
 | 
						|
            fullname = os.path.join(path, name)
 | 
						|
            displayname = linkname = name
 | 
						|
            # Append / for directories or @ for symbolic links
 | 
						|
            if os.path.isdir(fullname):
 | 
						|
                displayname = name + "/"
 | 
						|
                linkname = name + "/"
 | 
						|
            if os.path.islink(fullname):
 | 
						|
                displayname = name + "@"
 | 
						|
                # Note: a link to a directory displays with @ and links with /
 | 
						|
            f.write('<li><a href="%s">%s</a>\n'
 | 
						|
                    % (quote(linkname), cgi.escape(displayname)))
 | 
						|
        f.write("</ul>\n<hr>\n</body>\n</html>\n")
 | 
						|
        length = f.tell()
 | 
						|
        f.seek(0)
 | 
						|
        self.send_response(200)
 | 
						|
        encoding = sys.getfilesystemencoding()
 | 
						|
        self.send_header("Content-type", "text/html; charset=%s" % encoding)
 | 
						|
        self.send_header("Content-Length", str(length))
 | 
						|
        self.end_headers()
 | 
						|
        return f
 | 
						|
 | 
						|
    def translate_path(self, path):
 | 
						|
        """ Override to handle redirects.
 | 
						|
        """
 | 
						|
        path = path.split('?', 1)[0]
 | 
						|
        path = path.split('#', 1)[0]
 | 
						|
        path = normpath(unquote(path))
 | 
						|
        words = path.split('/')
 | 
						|
        words = filter(None, words)
 | 
						|
        path = self.serve_path
 | 
						|
        for word in words:
 | 
						|
            drive, word = splitdrive(word)
 | 
						|
            head, word = split(word)
 | 
						|
            if word in (curdir, pardir): continue
 | 
						|
            path = join(path, word)
 | 
						|
        return path
 | 
						|
 | 
						|
    # Private interface ######################################################
 | 
						|
 | 
						|
    def _get_range_header(self):
 | 
						|
        """ Returns request Range start and end if specified.
 | 
						|
        If Range header is not specified returns (None, None)
 | 
						|
        """
 | 
						|
 | 
						|
        if sys.version_info[0] < 3:
 | 
						|
            range_header = self.headers.getheader("Range")
 | 
						|
        else:
 | 
						|
            range_header = self.headers.get("Range")
 | 
						|
 | 
						|
        if range_header is None:
 | 
						|
            return (None, None)
 | 
						|
        if not range_header.startswith("bytes="):
 | 
						|
            return (None, None)
 | 
						|
        regex = re.compile(r"^bytes=(\d+)\-(\d+)?")
 | 
						|
        rangething = regex.search(range_header)
 | 
						|
        if rangething:
 | 
						|
            from_val = int(rangething.group(1))
 | 
						|
            if rangething.group(2) is not None:
 | 
						|
                return (from_val, int(rangething.group(2)))
 | 
						|
            else:
 | 
						|
                return (from_val, None)
 | 
						|
        else:
 | 
						|
            return (None, None)
 | 
						|
 | 
						|
 | 
						|
def get_server(port=8000, next_attempts=0, serve_path=None):
 | 
						|
    Handler = RequestHandler
 | 
						|
    if serve_path:
 | 
						|
        Handler.serve_path = serve_path
 | 
						|
    while next_attempts >= 0:
 | 
						|
        try:
 | 
						|
            httpd = ThreadingHTTPServer(("", port), Handler)
 | 
						|
            return httpd
 | 
						|
        except socket.error as e:
 | 
						|
            if e.errno == errno.EADDRINUSE:
 | 
						|
                next_attempts -= 1
 | 
						|
                port += 1
 | 
						|
            else:
 | 
						|
                raise
 | 
						|
 | 
						|
 | 
						|
def main(args=None):
 | 
						|
    if args is None:
 | 
						|
        args = sys.argv[1:]
 | 
						|
 | 
						|
    PORT = 8000
 | 
						|
    if len(args) > 0:
 | 
						|
        PORT = int(args[-1])
 | 
						|
    serve_path = DATA_DIR
 | 
						|
    if len(args) > 1:
 | 
						|
        serve_path = abspath(args[-2])
 | 
						|
 | 
						|
    httpd = get_server(port=PORT, serve_path=serve_path)
 | 
						|
 | 
						|
    print("serving at port " + str(PORT))
 | 
						|
    httpd.serve_forever()
 | 
						|
 | 
						|
 | 
						|
if __name__ == "__main__":
 | 
						|
    main()
 |