1.浏览器请求动态页面过程
2.WSGI
Python Web Server Gateway Interface (或简称 WSGI,读作“wizgy”)。
WSGI允许开发者将选择web框架和web服务器分开。可以混合匹配web服务器和web框架,选择一个适合的配对。比如,可以在Gunicorn 或者 Nginx/uWSGI 或者 Waitress上运行 Django, Flask, 或 Pyramid。真正的混合匹配,得益于WSGI同时支持服务器和架构.
web服务器必须具备WSGI接口,所有的现代Python Web框架都已具备WSGI接口,它让你不对代码作修改就能使服务器和特点的web框架协同工作。
WSGI由web服务器支持,而web框架允许你选择适合自己的配对,但它同样对于服务器和框架开发者提供便利使他们可以专注于自己偏爱的领域和专长而不至于相互牵制。其他语言也有类似接口:java有Servlet API,Ruby 有 Rack。
3.定义WSGI接口
WSGI接口定义非常简单,它只要求Web开发者实现一个函数,就可以响应HTTP请求。我们来看一个最简单的Web版本的“Hello World!”:
defapplication(environ, start_response): start_response('200 OK', [('Content-Type','text/html')])return'Hello World!'
上面的application( )函数就是符合WSGI标准的一个HTTP处理函数,它接收两个参数:
environ:一个包含所有HTTP请求信息的dict对象;
start_response:一个发送HTTP响应的函数。
整个application( )函数本身没有涉及到任何解析HTTP的部分,也就是说,把底层web服务器解析部分和应用程序逻辑部分进行了分离,这样开发者就可以专心做一个领域了.
application( )函数必须由WSGI服务器来调用。有很多符合WSGI规范的服务器。而我们此时的web服务器项目的目的就是做一个极可能解析静态网页还可以解析动态网页的服务器
实现代码:
importtime,multiprocessing,socket,os,reclassMyHttpServer(object):def__init__(self):serveSocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) self.serveSocket = serveSocket self.HTMLPATH ='./html'defbind(self,port=8000):self.serveSocket.bind(('',port))defstart(self):self.serveSocket.listen()whileTrue: clientSocket, clientAddr = self.serveSocket.accept() print(clientSocket) multiprocessing.Process(target=self.serveHandler, args=(clientSocket, clientAddr)).start() clientSocket.close()defserveHandler(self,clientSocket,clientAddr):try: recvData = clientSocket.recv(1024).decode('gbk') fileName = re.split(r' +', recvData.splitlines()[0])[1] filePath = self.HTMLPATHiffileName.endswith('.py'):try: pyname=fileName[1:-3]# 导入pyModule = __import__(pyname) env={} responseBody = pyModule.application(env,self.startResponse) responseLine = self.responseLine responseHeader = self.responseHeaderexceptImportError: responseLine ='HTTP/1.1 404 NOT FOUND'responseHeader ='Server: ererbai'+ os.linesep responseHeader +='Date: %s'% time.ctime() responseBody ='
很抱歉,服务器中找不到你想要的内容
'else:if'/'== fileName: filePath +='/index.html'else: filePath += fileNametry: file =Nonefile =open(filePath,'r',encoding='gbk') responseBody = file.read() responseLine ='HTTP/1.1 200 OK'responseHeader ='Server: ererbai'+ os.linesep responseHeader +='Date:%s'% time.ctime()exceptFileNotFoundError: responseLine ='HTTP/1.1 404 NOT FOUND'responseHeader ='Server: ererbai'+ os.linesep responseHeader +='Date:%s'% time.ctime() responseBody ='很抱歉,服务器中找不到你想要的内容'finally:if(file!=None)and(notfile.closed): file.close()exceptExceptionasex: responseLine ='HTTP/1.1 500 ERROR'responseHeader ='Server: ererbai'+ os.linesep responseHeader +='Date: %s'% time.ctime() responseBody ='服务器正在维护中,请稍后再试。%s'%exfinally: senData = responseLine + os.linesep + responseHeader + os.linesep + os.linesep + responseBody print(senData) senData = senData.encode('gbk') clientSocket.send(senData)if(clientSocket!=None)and(notclientSocket._closed): clientSocket.close()defstartResponse(self,status,responseHeaders):self.responseLine = status self.responseHeader =''fork,vinresponseHeaders: kv = k +':'+ v + os.linesep self.responseHeader += kvif__name__ =='__main__': server = MyHttpServer() server.bind(8000) server.start()
服务器中存在的html的文件:
index.html
首页-毕业季我们仍需共生命的慷慨与繁华相爱,即使岁月以刻薄和荒芜相欺。