10 分离式web框架

10 分离式web框架

wsgiref模块:

 http请求封装成以键值对的形式封装成字典environ

"PATH_INFO"对应的值为请求文件路径,

      QUERY_STRING”对应的值为get请求发送的数据

     REQUEST+METHOD”对应的值为请求方式。

同时封装一个发送响应格式的函数start_server通过特定的方式return [ ]返回数据并调用内置方法返回给浏览器。

 

GETPOST请求:

请求参数的获取:

    GET请求参数的传递在url请求文件路径后边拼接显式传输,可以直接在wsgiref模块封装好的environ请求信息字典中以键“QUERY_STRING”取值获取传递的参数;

   POST请求参数的传递是在请求信息的请求数据部分,

通过在wsgiref模块封装好的environ请求信息字典中以键“CONTENT_LENGTH”获取参数(bytes类型)长度,然后通过environ["wsgi.input"].read(长度)获取传递的参数(bytes类型)

请求参数的处理:

内置模块(from urllib.parse import parse_qs

执行parse_qs()进行格式化成字典找到对应的参数值(列表形式,注意get为字符串,post为字节);

 

具体处理方式:

GET:

query_string=environ["QUERY_STRING"]

 #'QUERY_STRING': 'username=zhangsan&userpsd=abc',

query_string=parse_qs(query_string)

#{ 'username':['zhangsan'],'userpsd':['abc']}

name=query_request["username"][0]

'zhangsan'

psd=query_request["userpsd"][0]

'abc'

POST:

leng=int(environ.get('CONTENT_LENGTH',0))

 #'CONTENT_LENGTH'参数长度

query_string=environ['wsgi.input'].read(leng)

#{ 'username':['zhangsan'],'userpsd':['abc']}

query_string=parse_qs(query_string)

#{ b'username':[b'zhangsan'],b'userpsd':[b'abc']}

name=query_string[b'username'][0].decode("utf-8")

'zhangsan'

psd=query_string[b'userpsd'][0].decode("utf-8")

'abc'

 

wsgiref封装的请求信息:

GET

{……. 'SERVER_PORT': '8888', 'REMOTE_HOST': '', 'CONTENT_LENGTH': '', 'SCRIPT_NAME': '', 'SERVER_PROTOCOL': 'HTTP/1.1', 'SERVER_SOFTWARE': 'WSGIServer/0.2', 'REQUEST_METHOD': 'GET', 'PATH_INFO': '/auth', 'QUERY_STRING': 'username=zhangsan&userpsd=abc', 'REMOTE_ADDR': '127.0.0.1', 'CONTENT_TYPE': 'text/plain', 'HTTP_HOST': '127.0.0.1:8888',  'HTTP_ACCEPT_LANGUAGE': 'zh-CN,zh;q=0.9', 'wsgi.input': <_io.BufferedReader name=916>…….}

POST

{…… 'GATEWAY_INTERFACE': 'CGI/1.1', 'SERVER_PORT': '8888', 'REMOTE_HOST': '', 'CONTENT_LENGTH': '29', 'SCRIPT_NAME': '', 'SERVER_PROTOCOL': 'HTTP/1.1', 'REQUEST_METHOD': 'POST', 'PATH_INFO': '/auth', 'QUERY_STRING': '',  'HTTP_ACCEPT_LANGUAGE': 'zh-CN,zh;q=0.9', 'wsgi.input': <_io.BufferedReader name=880>, 'wsgi.errors': <_io.TextIOWrapper name='<stderr>' mode='w' encoding='UTF-8'>, 'wsgi.version': (1, 0), 'wsgi.run_once': False, 'wsgi.url_scheme': 'http', 'wsgi.multithread': True, 'wsgi.multiprocess': False, 'wsgi.file_wrapper': <class 'wsgiref.util.FileWrapper'>} 


客户端浏览器访问:

    在浏览器地址栏直接输入地址端口:127.0.0.18888

 

服务器serverpython程序(分离版本):

 


 

manege.py: 

 1 from wsgiref.simple_server import make_server   #python内置基于socket的服务程序模块
 2 from urls import func_mappers       #自己封装的请求文件路径与对应操作映射模块
 3  
 4 def application(environ,start_reponse):
 5     """
 6     基于wsgiref模块实现客户端请求数据的封装(字典environ)和封装的响应格式发送函数(statr_response)进行服务响应
 7     :param environ: wsgiref模块封装好的请求信息字典
 8     :param start_reponse: wsgiref模块封装的响应格式发送函数
 9     :return:
10     """
11     path=environ["PATH_INFO"]       #通过字典取值获取请求路径信息
12     start_reponse("200 Ok",[("Content","text/html")])   #分装好的函数发送响应格式信息
13     # print(path)
14     # 根据请求文件路径,映射与之对应的函数操作
15     for mapper in func_mappers:
16         if path==mapper[0]:
17             data=mapper[1](environ)
18     else:
19         data=b"Sorry:404 not find"
20     return [data]
21  
22 if __name__ == '__main__':
23     httpd=make_server("127.0.0.1",8888,application)
24     httpd.serve_forever()
manage.py

 modles.py 

 1 # 连接数据库,创建需要实用的信息表
 2 import pymysql
 3 conn=pymysql.connect("127.0.0.1","root","","web")
 4 cursor=conn.cursor()
 5  
 6 user=[("zhangsan","abc"),
 7       ("lisi","123")]
 8 sql_createTable="create table userinfo(uid int primary key auto_increment,name char(12) unique not null ,password char(32) not null )"
 9 sql_insret="insert into userinfo(name,password) values"
10  
11 cursor.execute(sql_createTable)
12  
13 for  data in user:
14     sql_value=sql_insret+str(data)
15     cursor.execute(sql_value)
16 conn.commit()
17  
18 cursor.close()
19 conn.close()
models.py

 urls.py 

1 # 路由信息:请求文件路径与对应操作函数的映射表
2 from views import *
3 func_mappers=[
4     ("/",func_login),
5     ("/auth",func_auth)]
uels.py

views.py 

 1 from urllib.parse import parse_qs    #内置模块方法用于处理请求传递的数据格式
 2 from webauth import auth
 3  
 4 # 客户端请求根目录时响应登录页面
 5 def func_login(environ):
 6     with open("login.html", "rb")as f:
 7         data = f.read()
 8     return data
 9  
10 # 客户端提交登录时,判断请求方式,获取提交的数据,进行验证
11 def func_auth(environ):
12  
13     if environ.get("REQUEST_METHOD") == "GET":
14         query_request = environ["QUERY_STRING"]
15         # print(user)
16         # print(environ)
17  
18         query_request = parse_qs(query_request)
19         print(repr(query_request))
20         username = query_request["username"][0]
21         psd = query_request["userpsd"][0]
22  
23     elif environ.get("REQUEST_METHOD") == "POST":
24         # 获取请求体数据的长度,因为提交过来的数据需要用它来提取,注意POST请求和GET请求的获取数据的方式不同
25         try:
26             request_body_size = int(environ.get('CONTENT_LENGTH', 0))
27         except (ValueError):
28             request_body_size = 0
29         # POST请求获取数据的方式
30         request_data = environ['wsgi.input'].read(request_body_size)
31         print('>>>>>', request_data)  # >>>>> b'username=chao&password=123',是个bytes类型数据
32         print('?????', environ['QUERY_STRING'])  # ????? 空的,因为post请求只能按照上面这种方式取数据
33         # parse_qs可以帮我们解析数据
34         re_data = parse_qs(request_data)
35         print('拆解后的数据', re_data)  # 拆解后的数据 {b'password': [b'123'], b'username': [b'chao']}
36         username=re_data[b'username'][0].decode("utf-8")
37         psd=re_data[b'userpsd'][0].decode("utf-8")
38  
39     # 获取用户名和密码之后进行提交验证
40     status = auth(username, psd)
41     if status:
42         with open("login_success.html", "rb")as f:
43             data = f.read()
44     else:
45         data = "<h1> login error</h1>".encode("utf-8")
46     return data
views.py

 webauth.py 

 1 # 登录认证使用:连接数据库查询用户名和密码进行检验
 2 def auth(username,psd):
 3     import pymysql
 4     conn = pymysql.connect("127.0.0.1", "root", "", "web")
 5     cursor = conn.cursor()
 6     sql_select=f"select * from userinfo where name='{username}' and password='{psd}' "
 7     # print(sql_select)
 8     ret=cursor.execute(sql_select)  #函数返回结果ret为匹配到的记录条数
 9     cursor.close()
10     conn.cursor()
11     return ret
12 # print(auth('zhangsan','abc'))
webauth.py

login.html

 1 <!DOCTYPE html>
 2 <html lang="en">
 3 <head>
 4     <meta charset="UTF-8">     
 5     <link rel="icon" href="faviron.ico">
 6     <title>登录</title>
 7 </head>
 8 <body>
 9     <div>
10  
11         <form action="auth" method="post" name="login"><!--以post请求方式auth验证-->
12  
13         <!--<form action="auth" method="get" name="login">&lt;!&ndash;以get请求方式auth验证&ndash;&gt;-->
14             <table>
15                 <tr>
16                     <td><label for="username">用户名</label></td>
17                     <td><input type="text" name="username" id="username"></td>
18                 </tr>
19                 <tr>
20                     <td> <label for="userpsd">密码</label></td>
21                     <td><input type="password"  name="userpsd" id="userpsd"></td>
22                 </tr>
23                 <tr>
24                     <td ><input type="reset"></td>
25                     <td ><input type="submit"></td>
26                 </tr>
27             </table>
28  
29         </form>
30     </div>
31 </body>
32 <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.js"></script>
33 <script>
34     $(function () {
35         //在登录之前在客户端进行账号、密码初级判断
36         $("form[name='login']").submit (function () {
37             var name = $("#username").val();
38             var psd = $("#userpsd").val();
39  
40             if (name == "" || psd == "") {
41                 alert("用户名和密码不能为空!");
42                 return false
43             }
44         })
45     })
46 </script>
47 </html>
login.html

login_success.html 

 1 <!DOCTYPE html>
 2 <html lang="en">
 3 <head>
 4     <meta charset="UTF-8">
 5     <meta http-equiv="refresh" content="">
 6     <meta name="keywords" content="">
 7     <style></style>
 8     <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.js"></script>
 9     <link rel="stylesheet" href="">
10     <title>首页</title>
11 </head>
12 <body>
13     <div>
14         <h1>登录成功!</h1>
15     </div>
16 </body>
17 </html>
login_success.html

 

posted @ 2019-07-14 00:53  笑得好美  阅读(276)  评论(0编辑  收藏  举报