#!/usr/bin/env python3 import os, sys import os.path, time import posixpath import http.server import socketserver import urllib.request, urllib.parse, urllib.error import html import shutil import mimetypes import re import argparse import base64 from io import BytesIO def fbytes(B): B = float(B) KB = float(1024) MB = float(KB ** 2) # 1,048,576 GB = float(KB ** 3) # 1,073,741,824 TB = float(KB ** 4) # 1,099,511,627,776 if B < KB: return '{0} {1}'.format(B,'Bytes' if 0 == B > 1 else 'Byte') elif KB <= B < MB: return '{0:.2f} KB'.format(B/KB) elif MB <= B < GB: return '{0:.2f} MB'.format(B/MB) elif GB <= B < TB: return '{0:.2f} GB'.format(B/GB) elif TB <= B: return '{0:.2f} TB'.format(B/TB) class SimpleHTTPRequestHandler(http.server.BaseHTTPRequestHandler): def do_GET(self): f = self.send_head() if f: self.copyfile(f, self.wfile) f.close() def do_HEAD(self): f = self.send_head() if f: f.close() def do_POST(self): r, info = self.deal_post_data() print((r, info, "by: ", self.client_address)) f = BytesIO() f.write(b'') f.write(b"\nUpload Result Page\n") f.write(b'\n') f.write(b"\n

Upload Result Page

\n") f.write(b"
\n") if r: f.write(b"Success!") else: f.write(b"Failed!") f.write(info.encode()) f.write(("

" % self.headers['referer']).encode()) f.write(b"\n") f.write(b"here.\n\n") length = f.tell() f.seek(0) self.send_response(200) self.send_header("Content-type", "text/html") self.send_header("Content-Length", str(length)) self.end_headers() if f: self.copyfile(f, self.wfile) f.close() def deal_post_data(self): uploaded_files = [] content_type = self.headers['content-type'] if not content_type: return (False, "Content-Type header doesn't contain boundary") boundary = content_type.split("=")[1].encode() remainbytes = int(self.headers['content-length']) line = self.rfile.readline() remainbytes -= len(line) if not boundary in line: return (False, "Content NOT begin with boundary") while remainbytes > 0: line = self.rfile.readline() remainbytes -= len(line) fn = re.findall(r'Content-Disposition.*name="file"; filename="(.*)"', line.decode()) if not fn: return (False, "Can't find out file name...") path = self.translate_path(self.path) fn = os.path.join(path, fn[0]) line = self.rfile.readline() remainbytes -= len(line) line = self.rfile.readline() remainbytes -= len(line) try: out = open(fn, 'wb') except IOError: return (False, "

Can't create file to write.
Do you have permission to write?") else: with out: preline = self.rfile.readline() remainbytes -= len(preline) while remainbytes > 0: line = self.rfile.readline() remainbytes -= len(line) if boundary in line: preline = preline[0:-1] if preline.endswith(b'\r'): preline = preline[0:-1] out.write(preline) uploaded_files.append(fn) break else: out.write(preline) preline = line return (True, "

'%s'" % "'
'".join(uploaded_files)) def send_head(self): path = self.translate_path(self.path) f = None if os.path.isdir(path): if not self.path.endswith('/'): self.send_response(301) self.send_header("Location", self.path + "/") self.end_headers() return None for index in "index.html", "index.htm": index = os.path.join(path, index) if os.path.exists(index): path = index break else: return self.list_directory(path) ctype = self.guess_type(path) try: f = open(path, 'rb') except IOError: self.send_error(404, "File not found") return None self.send_response(200) self.send_header("Content-type", ctype) fs = os.fstat(f.fileno()) self.send_header("Content-Length", str(fs[6])) self.send_header("Last-Modified", self.date_time_string(fs.st_mtime)) self.end_headers() return f def list_directory(self, path): try: list = os.listdir(path) except os.error: self.send_error(404, "No permission to list directory") return None enc = sys.getfilesystemencoding() list.sort(key=lambda a: a.lower()) f = BytesIO() displaypath = html.escape(urllib.parse.unquote(self.path)) f.write(b'') f.write(b'\n') f.write(('' % enc).encode(enc)) f.write(("Directory listing for %s\n" % displaypath).encode(enc)) f.write(b'\n') f.write(("\n

Directory listing for %s

\n" % displaypath).encode(enc)) f.write(b"
\n") f.write(b"
") f.write(b"") f.write(b"
\n") f.write(b"
\n") f.write(b'\n') f.write(b'\n') for name in list: dirimage = '' fullname = os.path.join(path, name) displayname = linkname = name fsize = fbytes(os.path.getsize(fullname)) created_date = time.ctime(os.path.getctime(fullname)) # Append / for directories or @ for symbolic links if os.path.isdir(fullname): dirimage = '' displayname = name + "/" linkname = name + "/" fsize = '' created_date = '' if os.path.islink(fullname): dirimage = '' displayname = name + "@" if name.endswith(('.bmp','.gif','.jpg','.png')): dirimage = name if name.endswith(('.avi','.mpg')): dirimage = '' if name.endswith(('.idx','.srt','.sub')): dirimage = '' if name.endswith('.iso'): dirimage = '' f.write(('\n' % ( dirimage, urllib.parse.quote(linkname), html.escape(displayname) , fsize , created_date )).encode(enc)) f.write(b"
[PARENTDIR]Parent Directory
%s%s%s

\n\n\n") length = f.tell() f.seek(0) self.send_response(200) self.send_header("Content-type", "text/html") self.send_header("Content-Length", str(length)) self.end_headers() return f def translate_path(self, path): path = path.split('?',1)[0] path = path.split('#',1)[0] path = posixpath.normpath(urllib.parse.unquote(path)) words = path.split('/') words = [_f for _f in words if _f] path = os.getcwd() for word in words: drive, word = os.path.splitdrive(word) head, word = os.path.split(word) if word in (os.curdir, os.pardir): continue path = os.path.join(path, word) return path def copyfile(self, source, outputfile): shutil.copyfileobj(source, outputfile) def guess_type(self, path): base, ext = posixpath.splitext(path) if ext in self.extensions_map: return self.extensions_map[ext] ext = ext.lower() if ext in self.extensions_map: return self.extensions_map[ext] else: return self.extensions_map[''] if not mimetypes.inited: mimetypes.init() # try to read system mime.types extensions_map = mimetypes.types_map.copy() extensions_map.update({ '': 'application/octet-stream', # Default '.py': 'text/plain', '.c': 'text/plain', '.h': 'text/plain', }) parser = argparse.ArgumentParser() parser.add_argument('--bind', '-b', default='', metavar='ADDRESS', help='Specify alternate bind address [default: all interfaces]') parser.add_argument('port', action='store', default=8000, type=int, nargs='?', help='Specify alternate port [default: 8000]') args = parser.parse_args() PORT = args.port BIND = args.bind HOST = BIND if HOST == '': HOST = 'localhost' Handler = SimpleHTTPRequestHandler with socketserver.TCPServer((BIND, PORT), Handler) as httpd: serve_message = "Serving HTTP on {host} port {port} (http://{host}:{port}/) ..." print(serve_message.format(host=HOST, port=PORT)) httpd.serve_forever()