├── HTTPServerWithUpload.py └── README.md /HTTPServerWithUpload.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python3 2 | 3 | import os, sys 4 | import os.path, time 5 | import posixpath 6 | import http.server 7 | import socketserver 8 | import urllib.request, urllib.parse, urllib.error 9 | import html 10 | import shutil 11 | import mimetypes 12 | import re 13 | import argparse 14 | import base64 15 | 16 | from io import BytesIO 17 | 18 | def fbytes(B): 19 | B = float(B) 20 | KB = float(1024) 21 | MB = float(KB ** 2) # 1,048,576 22 | GB = float(KB ** 3) # 1,073,741,824 23 | TB = float(KB ** 4) # 1,099,511,627,776 24 | 25 | if B < KB: 26 | return '{0} {1}'.format(B,'Bytes' if 0 == B > 1 else 'Byte') 27 | elif KB <= B < MB: 28 | return '{0:.2f} KB'.format(B/KB) 29 | elif MB <= B < GB: 30 | return '{0:.2f} MB'.format(B/MB) 31 | elif GB <= B < TB: 32 | return '{0:.2f} GB'.format(B/GB) 33 | elif TB <= B: 34 | return '{0:.2f} TB'.format(B/TB) 35 | 36 | class SimpleHTTPRequestHandler(http.server.BaseHTTPRequestHandler): 37 | 38 | def do_GET(self): 39 | f = self.send_head() 40 | if f: 41 | self.copyfile(f, self.wfile) 42 | f.close() 43 | 44 | def do_HEAD(self): 45 | f = self.send_head() 46 | if f: 47 | f.close() 48 | 49 | def do_POST(self): 50 | r, info = self.deal_post_data() 51 | print((r, info, "by: ", self.client_address)) 52 | f = BytesIO() 53 | f.write(b'') 54 | f.write(b"\nUpload Result Page\n") 55 | f.write(b'\n') 59 | f.write(b"\n

Upload Result Page

\n") 60 | f.write(b"
\n") 61 | if r: 62 | f.write(b"Success!") 63 | else: 64 | f.write(b"Failed!") 65 | f.write(info.encode()) 66 | f.write(("

" % self.headers['referer']).encode()) 67 | f.write(b"\n") 68 | f.write(b"here.\n\n") 69 | length = f.tell() 70 | f.seek(0) 71 | self.send_response(200) 72 | self.send_header("Content-type", "text/html") 73 | self.send_header("Content-Length", str(length)) 74 | self.end_headers() 75 | if f: 76 | self.copyfile(f, self.wfile) 77 | f.close() 78 | 79 | def deal_post_data(self): 80 | uploaded_files = [] 81 | content_type = self.headers['content-type'] 82 | if not content_type: 83 | return (False, "Content-Type header doesn't contain boundary") 84 | boundary = content_type.split("=")[1].encode() 85 | remainbytes = int(self.headers['content-length']) 86 | line = self.rfile.readline() 87 | remainbytes -= len(line) 88 | if not boundary in line: 89 | return (False, "Content NOT begin with boundary") 90 | while remainbytes > 0: 91 | line = self.rfile.readline() 92 | remainbytes -= len(line) 93 | fn = re.findall(r'Content-Disposition.*name="file"; filename="(.*)"', line.decode()) 94 | if not fn: 95 | return (False, "Can't find out file name...") 96 | path = self.translate_path(self.path) 97 | fn = os.path.join(path, fn[0]) 98 | line = self.rfile.readline() 99 | remainbytes -= len(line) 100 | line = self.rfile.readline() 101 | remainbytes -= len(line) 102 | try: 103 | out = open(fn, 'wb') 104 | except IOError: 105 | return (False, "

Can't create file to write.
Do you have permission to write?") 106 | else: 107 | with out: 108 | preline = self.rfile.readline() 109 | remainbytes -= len(preline) 110 | while remainbytes > 0: 111 | line = self.rfile.readline() 112 | remainbytes -= len(line) 113 | if boundary in line: 114 | preline = preline[0:-1] 115 | if preline.endswith(b'\r'): 116 | preline = preline[0:-1] 117 | out.write(preline) 118 | uploaded_files.append(fn) 119 | break 120 | else: 121 | out.write(preline) 122 | preline = line 123 | return (True, "

'%s'" % "'
'".join(uploaded_files)) 124 | 125 | def send_head(self): 126 | path = self.translate_path(self.path) 127 | f = None 128 | if os.path.isdir(path): 129 | if not self.path.endswith('/'): 130 | self.send_response(301) 131 | self.send_header("Location", self.path + "/") 132 | self.end_headers() 133 | return None 134 | for index in "index.html", "index.htm": 135 | index = os.path.join(path, index) 136 | if os.path.exists(index): 137 | path = index 138 | break 139 | else: 140 | return self.list_directory(path) 141 | ctype = self.guess_type(path) 142 | try: 143 | f = open(path, 'rb') 144 | except IOError: 145 | self.send_error(404, "File not found") 146 | return None 147 | self.send_response(200) 148 | self.send_header("Content-type", ctype) 149 | fs = os.fstat(f.fileno()) 150 | self.send_header("Content-Length", str(fs[6])) 151 | self.send_header("Last-Modified", self.date_time_string(fs.st_mtime)) 152 | self.end_headers() 153 | return f 154 | 155 | 156 | 157 | def list_directory(self, path): 158 | try: 159 | list = os.listdir(path) 160 | except os.error: 161 | self.send_error(404, "No permission to list directory") 162 | return None 163 | enc = sys.getfilesystemencoding() 164 | list.sort(key=lambda a: a.lower()) 165 | f = BytesIO() 166 | displaypath = html.escape(urllib.parse.unquote(self.path)) 167 | f.write(b'') 168 | f.write(b'\n') 169 | f.write(('' % enc).encode(enc)) 171 | f.write(("Directory listing for %s\n" % displaypath).encode(enc)) 172 | f.write(b'\n') 182 | f.write(("\n

Directory listing for %s

\n" % displaypath).encode(enc)) 183 | f.write(b"
\n") 184 | f.write(b"
") 185 | f.write(b"") 186 | f.write(b"
\n") 187 | f.write(b"
\n") 188 | f.write(b'\n') 189 | f.write(b'\n') 190 | for name in list: 191 | dirimage = '' 192 | fullname = os.path.join(path, name) 193 | displayname = linkname = name 194 | fsize = fbytes(os.path.getsize(fullname)) 195 | created_date = time.ctime(os.path.getctime(fullname)) 196 | # Append / for directories or @ for symbolic links 197 | if os.path.isdir(fullname): 198 | dirimage = '' 199 | displayname = name + "/" 200 | linkname = name + "/" 201 | fsize = '' 202 | created_date = '' 203 | if os.path.islink(fullname): 204 | dirimage = '' 205 | displayname = name + "@" 206 | if name.endswith(('.bmp','.gif','.jpg','.png')): 207 | dirimage = name 208 | if name.endswith(('.avi','.mpg')): 209 | dirimage = '' 210 | if name.endswith(('.idx','.srt','.sub')): 211 | dirimage = '' 212 | if name.endswith('.iso'): 213 | dirimage = '' 214 | f.write(('\n' 215 | % ( dirimage, urllib.parse.quote(linkname), html.escape(displayname) , fsize , created_date )).encode(enc)) 216 | f.write(b"
[PARENTDIR]Parent Directory
%s%s%s

\n\n\n") 217 | length = f.tell() 218 | f.seek(0) 219 | self.send_response(200) 220 | self.send_header("Content-type", "text/html") 221 | self.send_header("Content-Length", str(length)) 222 | self.end_headers() 223 | return f 224 | 225 | def translate_path(self, path): 226 | path = path.split('?',1)[0] 227 | path = path.split('#',1)[0] 228 | path = posixpath.normpath(urllib.parse.unquote(path)) 229 | words = path.split('/') 230 | words = [_f for _f in words if _f] 231 | path = os.getcwd() 232 | for word in words: 233 | drive, word = os.path.splitdrive(word) 234 | head, word = os.path.split(word) 235 | if word in (os.curdir, os.pardir): continue 236 | path = os.path.join(path, word) 237 | return path 238 | 239 | def copyfile(self, source, outputfile): 240 | shutil.copyfileobj(source, outputfile) 241 | 242 | def guess_type(self, path): 243 | base, ext = posixpath.splitext(path) 244 | if ext in self.extensions_map: 245 | return self.extensions_map[ext] 246 | ext = ext.lower() 247 | if ext in self.extensions_map: 248 | return self.extensions_map[ext] 249 | else: 250 | return self.extensions_map[''] 251 | 252 | if not mimetypes.inited: 253 | mimetypes.init() # try to read system mime.types 254 | extensions_map = mimetypes.types_map.copy() 255 | extensions_map.update({ 256 | '': 'application/octet-stream', # Default 257 | '.py': 'text/plain', 258 | '.c': 'text/plain', 259 | '.h': 'text/plain', 260 | }) 261 | 262 | parser = argparse.ArgumentParser() 263 | parser.add_argument('--bind', '-b', default='', metavar='ADDRESS', help='Specify alternate bind address [default: all interfaces]') 264 | parser.add_argument('port', action='store', default=8000, type=int, nargs='?', help='Specify alternate port [default: 8000]') 265 | args = parser.parse_args() 266 | 267 | PORT = args.port 268 | BIND = args.bind 269 | HOST = BIND 270 | 271 | if HOST == '': 272 | HOST = 'localhost' 273 | 274 | Handler = SimpleHTTPRequestHandler 275 | 276 | with socketserver.TCPServer((BIND, PORT), Handler) as httpd: 277 | serve_message = "Serving HTTP on {host} port {port} (http://{host}:{port}/) ..." 278 | print(serve_message.format(host=HOST, port=PORT)) 279 | httpd.serve_forever() 280 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # HTTPServerWithUpload 2 | Python3 HTTP Server with upload functionality 3 | 4 | **Run server:** 5 | 6 | ``` 7 | python3 HTTPServerWithUpload.py 80 8 | ``` 9 | 10 | **Upload file:** 11 | 12 | * curl: 13 | 14 | ```bash 15 | curl -X POST http://server.com/ -H "Content-Type: multipart/form-data" -F file=@"file.zip" 16 | ``` 17 | 18 | * powershell: 19 | 20 | ```powershell 21 | $wc=New-Object System.Net.WebClient; $resp=$wc.UploadFile('http://server.com',"C:\file.zip") 22 | ``` --------------------------------------------------------------------------------