├── 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"\n
Upload Result Page\n")
55 | f.write(b'\n')
59 | f.write(b"\nUpload 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(("\nDirectory listing for %s
\n" % displaypath).encode(enc))
183 | f.write(b"
\n")
184 | f.write(b"\n")
187 | f.write(b"
\n")
188 | f.write(b'\n')
189 | f.write(b'![[PARENTDIR]]() | Parent Directory |
\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((' | %s | %s | %s |
\n'
215 | % ( dirimage, urllib.parse.quote(linkname), html.escape(displayname) , fsize , created_date )).encode(enc))
216 | f.write(b"
\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 | ```
--------------------------------------------------------------------------------