Python搭建http文件服务器实现手机电脑文件传输功能

第一种代码的界面如下:(有缺点,中文乱码)

# !/usr/bin/env python3
# -*- coding:utf-8 _*-"""Simple HTTP Server With Upload.
python -V3.6
This module builds on http.server by implementing the standard GET
and HEAD requests in a fairly straightforward manner.
"""__version__ = "0.1"
__all__ = ["SimpleHTTPRequestHandler"]
__author__ = "清纯世纪"
__home_page__ = "https://blog.csdn.net/qq_45100200/article/details/132343498"import os
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 base64from io import BytesIOclass SimpleHTTPRequestHandler(http.server.BaseHTTPRequestHandler):"""Simple HTTP request handler with GET/HEAD/POST commands.This serves files from the current directory and any of itssubdirectories.  The MIME type for files is determined bycalling the .guess_type() method. And can reveive file uploadedby client.The GET/HEAD/POST requests are identical except that the HEADrequest omits the actual contents of the file."""server_version = "SimpleHTTPWithUpload/" + __version__def do_GET(self):"""Serve a GET request."""f = self.send_head()if f:self.copyfile(f, self.wfile)f.close()def do_HEAD(self):"""Serve a HEAD request."""f = self.send_head()if f:f.close()def do_POST(self):"""Serve a POST request."""r, info = self.deal_post_data()print((r, info, "by: ", self.client_address))f = BytesIO()f.write(b'<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">')f.write(b"<html>\n<title>Upload Result Page</title>\n")f.write(b"<body>\n<h2>Upload Result Page</h2>\n")f.write(b"<hr>\n")if r:f.write(b"<strong>Success:</strong>")else:f.write(b"<strong>Failed:</strong>")f.write(info.encode())f.write(("<br><a href=\"%s\">back</a>" % self.headers['referer']).encode())f.write(b"<hr><small>Powered By: bones7456, check new version at ")f.write(b"<a href=\"https://gist.github.com/UniIsland/3346170\">")f.write(b"here</a>.</small></body>\n</html>\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)breakelse:out.write(preline)preline = linereturn (True, "File '%s' upload success!" % ",".join(uploaded_files))def send_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 copiedto the outputfile by the caller unless the command was HEAD,and must be closed by the caller under all circumstances), orNone, in which case the caller has nothing further to do."""path = self.translate_path(self.path)f = Noneif os.path.isdir(path):if not self.path.endswith('/'):# redirect browser - doing basically what apache doesself.send_response(301)self.send_header("Location", self.path + "/")self.end_headers()return Nonefor index in "index.html", "index.htm":index = os.path.join(path, index)if os.path.exists(index):path = indexbreakelse:return self.list_directory(path)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 Noneself.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 fdef list_directory(self, path):"""Helper to produce a directory listing (absent index.html).Return value is either a file object, or None (indicating anerror).  In either case, the headers are sent, making theinterface the same as for send_head()."""try:list = os.listdir(path)except os.error:self.send_error(404, "No permission to list directory")return Nonelist.sort(key=lambda a: a.lower())f = BytesIO()displaypath = html.escape(urllib.parse.unquote(self.path))f.write(b'<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">')f.write(("<html>\n<title>Directory listing for %s</title>\n" % displaypath).encode())f.write(b'<style type="text/css">\n')f.write(b'a { text-decoration: none; }\n')f.write(b'a:link { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:visited { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:active { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:hover { text-decoration: none; font-weight: bold; color: #ff0000; }\n')f.write(b'</style>\n')f.write(("<body>\n<h2>Directory listing for %s</h2>\n" % displaypath).encode())f.write(b"<hr>\n")f.write(b"<form ENCTYPE=\"multipart/form-data\" method=\"post\">")f.write(b"<input name=\"file\" type=\"file\" multiple/>")f.write(b"<input type=\"submit\" value=\"upload\"/></form>\n")f.write(b"<hr>\n")f.write(b'<a href="../"><img src="https://img-blog.csdnimg.cn/2022010703250844034.gif" alt="[PARENTDIR]" width="24" height="24">&nbsp;&nbsp;&nbsp;Parent Directory</a><br />\n')for name in list:dirimage = ''fullname = os.path.join(path, name)displayname = linkname = name# Append / for directories or @ for symbolic linksif os.path.isdir(fullname):dirimage = ''displayname = name + "/"linkname = name + "/"if os.path.islink(fullname):dirimage = ''displayname = name + "@"if name.endswith(('.bmp', '.gif', '.jpg', '.png')):dirimage = nameif name.endswith(('.avi', '.mpg')):dirimage = ''if name.endswith(('.idx', '.srt', '.sub')):dirimage = ''if name.endswith('.iso'):dirimage = ''# Note: a link to a directory displays with @ and links with /f.write(('<a href="%s"><img src="%s" width="24" height="24">&nbsp;&nbsp;&nbsp;%s</a><br />\n'% (urllib.parse.quote(linkname), dirimage, html.escape(displayname))).encode())f.write(b"<hr>\n</body>\n</html>\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 fdef translate_path(self, path):"""Translate a /-separated PATH to the local filename syntax.Components that mean special things to the local file system(e.g. drive or directory names) are ignored.  (XXX They shouldprobably be diagnosed.)"""# abandon query parameterspath = 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): continuepath = os.path.join(path, word)return pathdef copyfile(self, source, outputfile):"""Copy all data between two file objects.The SOURCE argument is a file object open for reading(or anything with a read() method) and the DESTINATIONargument is a file object open for writing (oranything with a write() method).The only reason for overriding this would be to changethe block size or perhaps to replace newlines by CRLF-- note however that this the default server uses thisto copy binary data as well."""shutil.copyfileobj(source, outputfile)def guess_type(self, path):"""Guess the type of a file.Argument is a PATH (a filename).Return value is a string of the form type/subtype,usable for a MIME Content-type header.The default implementation looks the file's extensionup in the table self.extensions_map, using application/octet-streamas a default; however it would be permissible (ifslow) to look inside the data to make a better guess."""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.typesextensions_map = mimetypes.types_map.copy()extensions_map.update({'': 'application/octet-stream',  # Default'.py': 'text/plain','.c': 'text/plain','.h': 'text/plain',})def get_local_ip():import socket"""获取本机IP地址"""try:s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)s.connect(('8.8.8.8', 80))ip = s.getsockname()[0]finally:s.close()return ipif __name__ == '__main__':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.portBIND = args.bindHOST = BINDif HOST == '':HOST = 'localhost'Handler = SimpleHTTPRequestHandlerwith 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))print("请输入以下地址进行文件传输:http://{}:{}".format(get_local_ip(),PORT))httpd.serve_forever()

第二种代码的界面如下:

#!/usr/bin/env python3"""Simple HTTP Server With Upload.
python -V3.6
This module builds on http.server by implementing the standard GET
and HEAD requests in a fairly straightforward manner.
"""__version__ = "0.1"
__all__ = ["SimpleHTTPRequestHandler"]
__author__ = "清纯世纪"
__home_page__ = "https://blog.csdn.net/qq_45100200/article/details/132343498"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 base64from io import BytesIOdef fbytes(B):'Return the given bytes as a human friendly KB, MB, GB, or TB string'B = float(B)KB = float(1024)MB = float(KB ** 2)  # 1,048,576GB = float(KB ** 3)  # 1,073,741,824TB = float(KB ** 4)  # 1,099,511,627,776if 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):"""Simple HTTP request handler with GET/HEAD/POST commands.This serves files from the current directory and any of itssubdirectories.  The MIME type for files is determined bycalling the .guess_type() method. And can reveive file uploadedby client.The GET/HEAD/POST requests are identical except that the HEADrequest omits the actual contents of the file."""server_version = "SimpleHTTPWithUpload/" + __version__def do_GET(self):"""Serve a GET request."""f = self.send_head()if f:self.copyfile(f, self.wfile)f.close()def do_HEAD(self):"""Serve a HEAD request."""f = self.send_head()if f:f.close()def do_POST(self):"""Serve a POST request."""r, info = self.deal_post_data()print((r, info, "by: ", self.client_address))f = BytesIO()f.write(b'<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">')f.write(b"<html>\n<title>Upload Result Page</title>\n")f.write(b'<style type="text/css">\n')f.write(b'* {font-family: Helvetica; font-size: 16px; }\n')f.write(b'a { text-decoration: none; }\n')f.write(b'</style>\n')f.write(b"<body>\n<h2>Upload Result Page</h2>\n")f.write(b"<hr>\n")if r:f.write(b"<strong>Success!</strong>")else:f.write(b"<strong>Failed!</strong>")f.write(info.encode())f.write(("<br><br><a href=\"%s\">" % self.headers['referer']).encode())f.write(b"<button>Back</button></a>\n")f.write(b"<hr><small>Powered By: bones7456<br>Check new version ")f.write(b"<a href=\"https://gist.github.com/UniIsland/3346170\" target=\"_blank\">")f.write(b"here</a>.</small></body>\n</html>\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, "<br><br>Can't create file to write.<br>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)breakelse:out.write(preline)preline = linereturn (True, "<br><br>'%s'" % "'<br>'".join(uploaded_files))def send_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 copiedto the outputfile by the caller unless the command was HEAD,and must be closed by the caller under all circumstances), orNone, in which case the caller has nothing further to do."""path = self.translate_path(self.path)f = Noneif os.path.isdir(path):if not self.path.endswith('/'):# redirect browser - doing basically what apache doesself.send_response(301)self.send_header("Location", self.path + "/")self.end_headers()return Nonefor index in "index.html", "index.htm":index = os.path.join(path, index)if os.path.exists(index):path = indexbreakelse:return self.list_directory(path)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 Noneself.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 fdef list_directory(self, path):"""Helper to produce a directory listing (absent index.html).Return value is either a file object, or None (indicating anerror).  In either case, the headers are sent, making theinterface the same as for send_head()."""try:list = os.listdir(path)except os.error:self.send_error(404, "No permission to list directory")return Noneenc = sys.getfilesystemencoding()list.sort(key=lambda a: a.lower())f = BytesIO()displaypath = html.escape(urllib.parse.unquote(self.path))f.write(b'<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">')f.write(b'<html>\n')f.write(('<meta http-equiv="Content-Type" ''content="text/html; charset=%s">' % enc).encode(enc))f.write(("<title>Directory listing for %s</title>\n" % displaypath).encode(enc))f.write(b'<style type="text/css">\n')f.write(b'* {font-family: Helvetica; font-size: 16px; }\n')f.write(b'a { text-decoration: none; }\n')f.write(b'a:link { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:visited { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:active { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:hover { text-decoration: none; font-weight: bold; color: #ff0000; }\n')f.write(b'table {\n  border-collapse: separate;\n}\n')f.write(b'th, td {\n  padding:0px 10px;\n}\n')f.write(b'</style>\n')f.write(("<body>\n<h2>Directory listing for %s</h2>\n" % displaypath).encode(enc))f.write(b"<hr>\n")f.write(b"<form ENCTYPE=\"multipart/form-data\" method=\"post\">")f.write(b"<input name=\"file\" type=\"file\" multiple/>")f.write(b"<input type=\"submit\" value=\"upload\"/></form>\n")f.write(b"<hr>\n")f.write(b'<table>\n')f.write(b'<tr><td><img src="" alt="[PARENTDIR]" width="24" height="24"></td><td><a href="../" >Parent Directory</a></td></tr>\n')for name in list:dirimage = ''fullname = os.path.join(path, name)displayname = linkname = namefsize = fbytes(os.path.getsize(fullname))created_date = time.ctime(os.path.getctime(fullname))# Append / for directories or @ for symbolic linksif 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 = nameif name.endswith(('.avi', '.mpg')):dirimage = ''if name.endswith(('.idx', '.srt', '.sub')):dirimage = ''if name.endswith('.iso'):dirimage = ''# Note: a link to a directory displays with @ and links with /f.write(('<tr><td><img src="%s" width="24" height="24"></td><td><a href="%s">%s</a></td><td style="text-align:right; font-weight: bold; color:#FF0000">%s</td><td style="text-align:right; font-weight: bold;">%s</td></tr>\n'% (dirimage, urllib.parse.quote(linkname), html.escape(displayname), fsize,created_date)).encode(enc))f.write(b"</table><hr>\n</body>\n</html>\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 fdef translate_path(self, path):"""Translate a /-separated PATH to the local filename syntax.Components that mean special things to the local file system(e.g. drive or directory names) are ignored.  (XXX They shouldprobably be diagnosed.)"""# abandon query parameterspath = 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): continuepath = os.path.join(path, word)return pathdef copyfile(self, source, outputfile):"""Copy all data between two file objects.The SOURCE argument is a file object open for reading(or anything with a read() method) and the DESTINATIONargument is a file object open for writing (oranything with a write() method).The only reason for overriding this would be to changethe block size or perhaps to replace newlines by CRLF-- note however that this the default server uses thisto copy binary data as well."""shutil.copyfileobj(source, outputfile)def guess_type(self, path):"""Guess the type of a file.Argument is a PATH (a filename).Return value is a string of the form type/subtype,usable for a MIME Content-type header.The default implementation looks the file's extensionup in the table self.extensions_map, using application/octet-streamas a default; however it would be permissible (ifslow) to look inside the data to make a better guess."""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.typesextensions_map = mimetypes.types_map.copy()extensions_map.update({'': 'application/octet-stream',  # Default'.py': 'text/plain','.c': 'text/plain','.h': 'text/plain',})def get_local_ip():import socket"""获取本机IP地址"""try:s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)s.connect(('8.8.8.8', 80))ip = s.getsockname()[0]finally:s.close()return ipif __name__ == '__main__':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.portBIND = args.bindHOST = BINDif HOST == '':HOST = 'localhost'Handler = SimpleHTTPRequestHandlerwith 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))print("请输入以下地址进行文件传输:http://{}:{}".format(get_local_ip(), PORT))httpd.serve_forever()

最后在来一个传输进度条显示的吧,基于第二个代码,第一个照着改就行。(也有缺点,我这使用后好像传输速度变慢了,可以自行测试验证):

#!/usr/bin/env python3"""Simple HTTP Server With Upload.
python -V3.6
This module builds on http.server by implementing the standard GET
and HEAD requests in a fairly straightforward manner.
"""__version__ = "0.1"
__all__ = ["SimpleHTTPRequestHandler"]
__author__ = "清纯世纪"
__home_page__ = "https://blog.csdn.net/qq_45100200/article/details/132343498"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 tqdm import tqdm
from io import BytesIOdef fbytes(B):'Return the given bytes as a human friendly KB, MB, GB, or TB string'B = float(B)KB = float(1024)MB = float(KB ** 2)  # 1,048,576GB = float(KB ** 3)  # 1,073,741,824TB = float(KB ** 4)  # 1,099,511,627,776if 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):"""Simple HTTP request handler with GET/HEAD/POST commands.This serves files from the current directory and any of itssubdirectories.  The MIME type for files is determined bycalling the .guess_type() method. And can reveive file uploadedby client.The GET/HEAD/POST requests are identical except that the HEADrequest omits the actual contents of the file."""server_version = "SimpleHTTPWithUpload/" + __version__def do_GET(self):"""Serve a GET request."""f = self.send_head()if f:self.copyfile(f, self.wfile)f.close()def do_HEAD(self):"""Serve a HEAD request."""f = self.send_head()if f:f.close()def do_POST(self):"""Serve a POST request."""r, info = self.deal_post_data()print((r, info, "by: ", self.client_address))f = BytesIO()f.write(b'<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">')f.write(b"<html>\n<title>Upload Result Page</title>\n")f.write(b'<style type="text/css">\n')f.write(b'* {font-family: Helvetica; font-size: 16px; }\n')f.write(b'a { text-decoration: none; }\n')f.write(b'</style>\n')f.write(b"<body>\n<h2>Upload Result Page</h2>\n")f.write(b"<hr>\n")if r:f.write(b"<strong>Success!</strong>")else:f.write(b"<strong>Failed!</strong>")f.write(info.encode())f.write(("<br><br><a href=\"%s\">" % self.headers['referer']).encode())f.write(b"<button>Back</button></a>\n")f.write(b"<hr><small>Powered By: bones7456<br>Check new version ")f.write(b"<a href=\"https://gist.github.com/UniIsland/3346170\" target=\"_blank\">")f.write(b"here</a>.</small></body>\n</html>\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, "<br><br>Can't create file to write.<br>Do you have permission to write?")else:with out:pbar = tqdm()  # 建立一个监听器preline = self.rfile.readline()remainbytes -= len(preline)total_size = remainbyteswhile remainbytes > 0:pbar.set_description("Processing {:.2f} %".format((1-(remainbytes/total_size))*100)) # 进度条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)breakelse:out.write(preline)preline = linereturn (True, "<br><br>'%s'" % "'<br>'".join(uploaded_files))def send_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 copiedto the outputfile by the caller unless the command was HEAD,and must be closed by the caller under all circumstances), orNone, in which case the caller has nothing further to do."""path = self.translate_path(self.path)f = Noneif os.path.isdir(path):if not self.path.endswith('/'):# redirect browser - doing basically what apache doesself.send_response(301)self.send_header("Location", self.path + "/")self.end_headers()return Nonefor index in "index.html", "index.htm":index = os.path.join(path, index)if os.path.exists(index):path = indexbreakelse:return self.list_directory(path)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 Noneself.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 fdef list_directory(self, path):"""Helper to produce a directory listing (absent index.html).Return value is either a file object, or None (indicating anerror).  In either case, the headers are sent, making theinterface the same as for send_head()."""try:list = os.listdir(path)except os.error:self.send_error(404, "No permission to list directory")return Noneenc = sys.getfilesystemencoding()list.sort(key=lambda a: a.lower())f = BytesIO()displaypath = html.escape(urllib.parse.unquote(self.path))f.write(b'<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">')f.write(b'<html>\n')f.write(('<meta http-equiv="Content-Type" ''content="text/html; charset=%s">' % enc).encode(enc))f.write(("<title>Directory listing for %s</title>\n" % displaypath).encode(enc))f.write(b'<style type="text/css">\n')f.write(b'* {font-family: Helvetica; font-size: 16px; }\n')f.write(b'a { text-decoration: none; }\n')f.write(b'a:link { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:visited { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:active { text-decoration: none; font-weight: bold; color: #0000ff; }\n')f.write(b'a:hover { text-decoration: none; font-weight: bold; color: #ff0000; }\n')f.write(b'table {\n  border-collapse: separate;\n}\n')f.write(b'th, td {\n  padding:0px 10px;\n}\n')f.write(b'</style>\n')f.write(("<body>\n<h2>Directory listing for %s</h2>\n" % displaypath).encode(enc))f.write(b"<hr>\n")f.write(b"<form ENCTYPE=\"multipart/form-data\" method=\"post\">")f.write(b"<input name=\"file\" type=\"file\" multiple/>")f.write(b"<input type=\"submit\" value=\"upload\"/></form>\n")f.write(b"<hr>\n")f.write(b'<table>\n')f.write(b'<tr><td><img src="" alt="[PARENTDIR]" width="24" height="24"></td><td><a href="../" >Parent Directory</a></td></tr>\n')for name in list:dirimage = ''fullname = os.path.join(path, name)displayname = linkname = namefsize = fbytes(os.path.getsize(fullname))created_date = time.ctime(os.path.getctime(fullname))# Append / for directories or @ for symbolic linksif 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 = nameif name.endswith(('.avi', '.mpg')):dirimage = ''if name.endswith(('.idx', '.srt', '.sub')):dirimage = ''if name.endswith('.iso'):dirimage = ''# Note: a link to a directory displays with @ and links with /f.write(('<tr><td><img src="%s" width="24" height="24"></td><td><a href="%s">%s</a></td><td style="text-align:right; font-weight: bold; color:#FF0000">%s</td><td style="text-align:right; font-weight: bold;">%s</td></tr>\n'% (dirimage, urllib.parse.quote(linkname), html.escape(displayname), fsize,created_date)).encode(enc))f.write(b"</table><hr>\n</body>\n</html>\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 fdef translate_path(self, path):"""Translate a /-separated PATH to the local filename syntax.Components that mean special things to the local file system(e.g. drive or directory names) are ignored.  (XXX They shouldprobably be diagnosed.)"""# abandon query parameterspath = 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): continuepath = os.path.join(path, word)return pathdef copyfile(self, source, outputfile):"""Copy all data between two file objects.The SOURCE argument is a file object open for reading(or anything with a read() method) and the DESTINATIONargument is a file object open for writing (oranything with a write() method).The only reason for overriding this would be to changethe block size or perhaps to replace newlines by CRLF-- note however that this the default server uses thisto copy binary data as well."""shutil.copyfileobj(source, outputfile)def guess_type(self, path):"""Guess the type of a file.Argument is a PATH (a filename).Return value is a string of the form type/subtype,usable for a MIME Content-type header.The default implementation looks the file's extensionup in the table self.extensions_map, using application/octet-streamas a default; however it would be permissible (ifslow) to look inside the data to make a better guess."""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.typesextensions_map = mimetypes.types_map.copy()extensions_map.update({'': 'application/octet-stream',  # Default'.py': 'text/plain','.c': 'text/plain','.h': 'text/plain',})def get_local_ip():import socket"""获取本机IP地址"""try:s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)s.connect(('8.8.8.8', 80))ip = s.getsockname()[0]finally:s.close()return ipif __name__ == '__main__':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.portBIND = args.bindHOST = BINDif HOST == '':HOST = 'localhost'Handler = SimpleHTTPRequestHandlerwith 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))print("请输入以下地址进行文件传输:http://{}:{}".format(get_local_ip(), PORT))httpd.serve_forever()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/48448.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

智能算法挑战赛决赛题目——初中组

题目 1. 判断是否存在重复的子序列 从 m 个字符中选取字符&#xff0c;生成 n 个符号的序列&#xff0c;使得其中没有 2 个相邻的子序列相同。如从 1&#xff0c;2&#xff0c;3&#xff0c;生成长度为 5 的序列&#xff0c;序列“12321”是合格的&#xff0c;而“12323”和“…

华为星闪,一项将 “ 更稳 WiFi ” 和 “ 更好蓝牙 ” 融合起来的通信标准

兼顾多用途和专业化的 AI 大模型、移除安卓代码的 HarmonyOS NEXT 、给折叠屏应用提供适配方向的《 折叠屏/平板应用体验评估标准 》。。。 不过除了这些比较贴近我们普通用户&#xff0c;容易讲清楚的东西&#xff0c;华为还官宣了一个大家可能没注意的黑科技&#xff1a; 星…

TCP性能机制

延迟应答 为什么有延迟应答 发送方如果长时间没有收到ACK应答&#xff0c;则会触发超时重传机制&#xff0c;重新发送数据包。但如果接收数据的主机立刻返回ACK应答, 这时候返回的窗口可能比较小&#xff0c;发送方一次只能发少量数据&#xff0c;效率较低。 举个例子理解一…

【深度学习-图像识别】使用fastai对Caltech101数据集进行图像多分类(50行以内的代码就可达到很高准确率)

文章目录 前言fastai介绍数据集介绍 一、环境准备二、数据集处理1.数据目录结构2.导入依赖项2.读入数据3.模型构建3.1 寻找合适的学习率3.2 模型调优 4.模型保存与应用 总结人工智能-图像识别 系列文章目录 前言 fastai介绍 fastai 是一个深度学习库&#xff0c;它为从业人员…

Spring Boot实践八--用户管理系统

一&#xff0c;技术介绍 技术选型功能说明springboot是一种基于 Spring 框架的快速开发应用程序的框架&#xff0c;它的主要作用是简化 Spring 应用程序的配置和开发&#xff0c;同时提供一系列开箱即用的功能和组件&#xff0c;如内置服务器、数据访问、安全、监控等&#xf…

[oneAPI] 基于BERT预训练模型的SWAG问答任务

[oneAPI] 基于BERT预训练模型的SWAG问答任务 基于Intel DevCloud for oneAPI下的Intel Optimization for PyTorch基于BERT预训练模型的SWAG问答任务数据集下载和描述数据集构建问答选择模型训练 结果参考资料 比赛&#xff1a;https://marketing.csdn.net/p/f3e44fbfe46c465f4d…

方案:AI边缘计算智慧工地解决方案

一、方案背景 在工程项目管理中&#xff0c;工程施工现场涉及面广&#xff0c;多种元素交叉&#xff0c;状况较为复杂&#xff0c;如人员出入、机械运行、物料运输等。特别是传统的现场管理模式依赖于管理人员的现场巡查。当发现安全风险时&#xff0c;需要提前报告&#xff0…

合宙Air724UG LuatOS-Air LVGL API--对象

对象 概念 在 LVGL 中&#xff0c;用户界面的基本构建块是对象。例如&#xff0c;按钮&#xff0c;标签&#xff0c;图像&#xff0c;列表&#xff0c;图表或文本区域。 属性 基本属性 所有对象类型都共享一些基本属性&#xff1a; Position (位置) Size (尺寸) Parent (父母…

ECMAScript6 简介及拓展

ECMAScript简介 JavaScript是大家所了解的语言名称&#xff0c; 但它的正式名称叫做ECMAScript。 1996年11月&#xff0c; JavaScript的创造者网景公司将JavaScript提交给国际化组织 ECMA(欧洲计算机制造联合会)&#xff0c; 希望这种语言能够成为国际标准。 随后 ECMA 发布…

【二叉树】572. 另一棵树的子树

572. 另一棵树的子树 解题思路 遍历二叉树的思路针对每一个节点判断该节点的子树和subtree是不是相等需要编写判断两个子树是否相等的函数 /*** Definition for a binary tree node.* public class TreeNode {* int val;* TreeNode left;* TreeNode right;* …

linux 免交互

Linux 免交互 1、免交互概念2、基本免交互的例子2.1命令行免交互统计2.2使用脚本免交互统计2.3使用免交互命令打印2.4免交互修改密码2.5重定向查看2.6重定向到指定文件2.7重定向直接指定文件2.8使用脚本完成重定向输入2.9免交互脚本完成赋值变量2.10关闭变量替换功能&#xff0…

[国产MCU]-W801开发实例-PWM控制器与LED亮度调节

PWM控制器与LED亮度调节 文章目录 PWM控制器与LED亮度调节1、PWM控制器2、PWM驱动API介绍3、PWM示例实现本文将详细介绍如何通过使用W801的PWM模块来控制LED的亮度。 1、PWM控制器 W801的PWM控制器具有如下主要特点: 5通道PWM信号生成功能2通道输入信号捕获功能(PWM0和PWM4两…

云计算在IT领域的发展和应用

文章目录 云计算的发展历程云计算的核心概念云计算在IT领域的应用1. 基础设施即服务&#xff08;IaaS&#xff09;&#xff1a;2. 平台即服务&#xff08;PaaS&#xff09;&#xff1a;3. 软件即服务&#xff08;SaaS&#xff09;&#xff1a; 云计算的拓展应用结论 &#x1f3…

华为OD-乱序数组两数之和绝对值最小

题目描述 给定一个随机的整数数组(可能存在正整数和负整数)nums, 请你在该数组中找出两个数&#xff0c;其和的绝对值(|nums[x]nums[y]|)为最小值 并返回这两个数(按从小到大返回)以及绝对值。 每种输入只会对应一个答案。但是&#xff0c;数组中同一个元素不能使用两遍。 输…

在Spring Boot应用程序中配置了两个不同的SOAP Web服务端点

新建一个CustomMessageDispatcherServlet类&#xff0c;以扩展以下MessageDispatcherServlet类&#xff0c; import org.springframework.ws.transport.http.MessageDispatcherServlet;import javax.servlet.http.HttpServletRequest;public class CustomMessageDispatcherSer…

如何进行在线pdf转ppt?在线pdf转ppt的方法

在当今数字化时代&#xff0c;PDF文件的广泛应用为我们的工作和学习带来了巨大的便利。然而&#xff0c;有时候我们可能需要将PDF转换为PPT文件&#xff0c;以便更好地展示和分享内容。在线PDF转PPT工具因其操作简便、高效而备受欢迎。如何进行在线pdf转ppt呢?接下来&#xff…

【Vue2.0源码学习】生命周期篇-初始化阶段(initLifecycle)

文章目录 1. 前言2. initLifecycle函数分析3. 总结 1. 前言 在上篇文章中&#xff0c;我们介绍了生命周期初始化阶段的整体工作流程&#xff0c;以及在该阶段都做了哪些事情。我们知道了&#xff0c;在该阶段会调用一些初始化函数&#xff0c;对Vue实例的属性、数据等进行初始…

fatal: not a git repository (or any of the parent directories): .git

提示说没有.git这样一个目录 在命令行 输入 git init 然后回车就好了 git remote add origin https:/.git git push -u origin "master"

《Java极简设计模式》第04章:建造者模式(Builder)

作者&#xff1a;冰河 星球&#xff1a;http://m6z.cn/6aeFbs 博客&#xff1a;https://binghe.gitcode.host 文章汇总&#xff1a;https://binghe.gitcode.host/md/all/all.html 源码地址&#xff1a;https://github.com/binghe001/java-simple-design-patterns/tree/master/j…

Node.js下载安装及环境配置教程

一、进入官网地址下载安装包 https://nodejs.org/zh-cn/download/ 选择对应你系统的Node.js版本&#xff0c;这里我选择的是Windows系统、64位 Tips&#xff1a;如果想下载指定版本&#xff0c;点击【以往的版本】&#xff0c;即可选择自己想要的版本下载 二、安装程序 &a…