午夜视频免费看_日韩三级电影网站_国产精品久久一级_亚洲一级在线播放_人妻体内射精一区二区三区_91夜夜揉人人捏人人添红杏_91福利在线导航_国产又粗又猛又黄又爽无遮挡_欧美日韩一区在线播放_中文字幕一区二区三区四区不卡 _日日夜夜精品视频免费观看_欧美韩日一区二区三区

主頁 > 知識庫 > python wsgiref源碼解析

python wsgiref源碼解析

熱門標簽:江門智能電話機器人 如何申請400電話代理 智能電話機器人調研 地圖標注可以遠程操作嗎 杭州房產地圖標注 滴滴地圖標注公司 甘肅高頻外呼系統 400電話在線如何申請 天津塘沽區地圖標注

python web開發中http請求的處理流程通常是: web-browser , web-server , wsgi 和 web-application四個環節, 我們學習過基于bottle實現的web-application,也學習了http.server。再完成python3源碼中自帶的wsgiref的庫,就可以拼接最后一個環節wsgi。本文會分下面幾個部分:

  • wsgi相關概念
  • cgi示例
  • wsgiref源碼
  • wsgi小結
  • 小技巧

wsgi 相關概念

CGI

CGI(Common Gateway Interface)通用網關接口。1993年由美國NCSA(National Center for Supercomputing Applications)發明。它具有簡單易用、語言無關的特點。雖然今天已經少有人直接使用CGI進行編程,但它仍被主流的Web服務器,如Apache、IIS、Nginx等廣泛支持。

CGI提供了一種接口規范,可以讓應用程序, 一般是各種腳本語言,比如perl, php, python等來擴展web服務,讓服務動態起來。

WSGI

WSGI(Web Server Gateway Interface)web服務網關接口。是web服務和web應用程序之間的接口規范,在PEP3333中提出。

wsgi讓應用程序和web服務之間解耦,應用程序只需要遵守規范,就可以在各種不同的web服務部署運行。比如上圖中,基于flask/django實現的應用程序可以使用gunicorn部署,也可以使用nginx+uwsgi部署。

ASGI

ASGI(Asynchronous Server Gateway Interface) 異步服務器網關接口。ASGI繼承自wsgi,旨在在具有異步功能的Python Web服務器,框架和應用程序之間提供標準接口。ASGI具有WSGI向后兼容性實現以及多個服務器和應用程序框架。

wsgi中使用請求響應模型,每個請求可以同步獲得一個響應。在ASGI中,請求的響應變成異步實現,一般用于websocket協議。(asgi的內容,涉及異步實現,本文就不多介紹)

cgi 示例

單純的概念理解比較難。下面我們配合示例一起來學習,先從CGI開始。

http 模塊提供了一個簡單的文件目錄服務:

python3 -m http.server
Serving HTTP on :: port 8000 (http://[::]:8000/) ...

這個服務只有靜態的展示功能,我們可以利用cgi擴展一個動態功能。

cgi腳本

創建cgi-bin目錄,這是CGI中約定的目錄名稱。然后編寫 hello.py, 代碼如下:

#!/usr/bin/env python

import time
import sqlite3
import os

DB_FILE = "guests.db"

def init_db():
	pass # 詳情請見附件

def update_total(ts):
	pass # 詳情請見附件

print('html>')
print('head>')
print('meta charset="utf-8">')
print('title>Hello Word!/title>')
print('/head>')
print('body>')
print('h2>Hello Python!/h2>')
if not os.path.exists(DB_FILE):
	init_db()
total = update_total(time.time())
print(f'total guest: {total}!')	
print('/body>')
print('/html>')

為了代碼簡潔,省略了db操作部分的具體實現。還需要給腳本可執行權限:

源碼在這里

chmod 755 hello.py

./hello.py
html>
head>
meta charset="utf-8">
title>Hello Word!/title>
/head>
body>
h2>Hello Python!/h2>
total guest: 4!
/body>
/html>

啟動http.server中的cgi服務:

python -m http.server --cgi

注意后面的 --cgi 參數,讓服務使用cgi-handler。啟動后使用 curl 訪問:

curl -v http://127.0.0.1:8000/cgi-bin/hello.py
* Trying 127.0.0.1...
* TCP_NODELAY set
* Connected to 127.0.0.1 (127.0.0.1) port 8000 (#0)
> GET /cgi-bin/hello.py HTTP/1.1
> Host: 127.0.0.1:8000
> User-Agent: curl/7.64.1
> Accept: */*
>
* HTTP 1.0, assume close after body
 HTTP/1.0 200 Script output follows
 Server: SimpleHTTP/0.6 Python/3.8.5
 Date: Sun, 31 Jan 2021 13:09:29 GMT
 html>
 head>
 meta charset="utf-8">
 title>Hello Word!/title>
 /head>
 body>
 h2>Hello Python!/h2>
 total guest: 5! # 訪客數
 /body>
 /html>
* Closing connection 0

可以看到 hello.py 正確執行,訪客數+1。因為數據存儲在db中,重啟服務仍然有效。

cgi服務實現

cgi的實現,主要就是下面的代碼:

# http.server

class CGIHTTPRequestHandler(SimpleHTTPRequestHandler):

 def run_cgi(self):
  import subprocess
  cmdline = [scriptfile]
  if self.is_python(scriptfile):
   interp = sys.executable
   cmdline = [interp, '-u'] + cmdline
  if '=' not in query:
   cmdline.append(query)

  try:
   nbytes = int(length)
  except (TypeError, ValueError):
   nbytes = 0
  p = subprocess.Popen(cmdline,
        stdin=subprocess.PIPE,
        stdout=subprocess.PIPE,
        stderr=subprocess.PIPE,
        env = env
        )
  if self.command.lower() == "post" and nbytes > 0:
   data = self.rfile.read(nbytes)
  # throw away additional data [see bug #427345]
  while select.select([self.rfile._sock], [], [], 0)[0]:
   if not self.rfile._sock.recv(1):
    break
  stdout, stderr = p.communicate(data)
  self.wfile.write(stdout)
  p.stderr.close()
  p.stdout.close()
  status = p.returncode

可見cgi的實現就是:

  • 使用subprocess.Popen新開了一個進程去執行腳本
  • 重定向腳本的輸出到當前socket的wfile,也就是http請求的返回上

代碼也驗證了為什么需要授予 hello.py 的可執行權限。

從例子可以了解到http.server專注于提供http服務,app.py專注于業務功能,兩者通過cgi進行銜接。

wsgiref

wsgiref是python自帶的wsgi的實現參考(reference), 主要代碼結構:

文件 描述
handlers.py wsgi實現
headers.py 管理http-header
simple_server.py 支持wsgi的http服務
util.pyvalidator.py 工具和驗證器

WSGIServer的代碼:

class WSGIServer(HTTPServer):

 """BaseHTTPServer that implements the Python WSGI protocol"""

 application = None

 def server_bind(self):
  """Override server_bind to store the server name."""
  HTTPServer.server_bind(self)
  self.setup_environ()

 def setup_environ(self): # 初始化環境變量
  # Set up base environment
  env = self.base_environ = {}
  env['SERVER_NAME'] = self.server_name
  env['GATEWAY_INTERFACE'] = 'CGI/1.1'
  env['SERVER_PORT'] = str(self.server_port)
  env['REMOTE_HOST']=''
  env['CONTENT_LENGTH']=''
  env['SCRIPT_NAME'] = ''

 def get_app(self):
  return self.application

 def set_app(self,application): # 注入application的class,注意是class
  self.application = application

WSGIServer并不復雜,繼承自http-server,接受application注入,就把web-server和we-application銜接起來。銜接后的動作,則是老規矩,交給HTTPRequestHandler去實現。同時wsgi服務多了一個準備env的動作,約定了一些wsgi的環境變量。

class WSGIRequestHandler(BaseHTTPRequestHandler):

 server_version = "WSGIServer/" + __version__

 def get_environ(self):
  pass

 def handle(self):
  """Handle a single HTTP request"""

  self.raw_requestline = self.rfile.readline(65537)
  if len(self.raw_requestline) > 65536:
   ...
   self.send_error(414)
   return

  if not self.parse_request(): # An error code has been sent, just exit
   return

  handler = ServerHandler(
   self.rfile, self.wfile, self.get_stderr(), self.get_environ(),
   multithread=False,
  ) # 創建新的業務handler
  handler.request_handler = self  
  handler.run(self.server.get_app()) # 創建application對象

WSGIRequestHandler覆蓋了handler,處理完成http協議(parse_request)后, 又做了四個動作:

  • 創建environ
  • 創建ServerHandler對象
  • 創建app對象
  • 運行app

environ處理主要是把http請求的header信息附帶在wsgi-server的環境變量上:

def get_environ(self):
 env = self.server.base_environ.copy() # wsgi-server的環境變量
 env['SERVER_PROTOCOL'] = self.request_version
 env['SERVER_SOFTWARE'] = self.server_version
 env['REQUEST_METHOD'] = self.command
 
 ...
 
 host = self.address_string()
 if host != self.client_address[0]:
  env['REMOTE_HOST'] = host
 env['REMOTE_ADDR'] = self.client_address[0]

 if self.headers.get('content-type') is None:
  env['CONTENT_TYPE'] = self.headers.get_content_type()
 else:
  env['CONTENT_TYPE'] = self.headers['content-type']

 length = self.headers.get('content-length')
 if length:
  env['CONTENT_LENGTH'] = length

 for k, v in self.headers.items():
  k=k.replace('-','_').upper(); v=v.strip()
  if k in env:
   continue     # skip content length, type,etc.
  if 'HTTP_'+k in env:
   env['HTTP_'+k] += ','+v  # comma-separate multiple headers
  else:
   env['HTTP_'+k] = v
 return env

ServerHandler對象的創建,接受輸入/輸出/錯誤,以及環境變量信息:

class ServerHandler(BaseHandler):

 def __init__(self,stdin,stdout,stderr,environ,
  multithread=True, multiprocess=False
 ):
  self.stdin = stdin
  self.stdout = stdout
  self.stderr = stderr
  self.base_env = environ
  self.wsgi_multithread = multithread
  self.wsgi_multiprocess = multiprocess
 ...

重點在ServerHandler的run函數:

class BaseHandler:
 def run(self, application):
 """Invoke the application"""
 # Note to self: don't move the close()! Asynchronous servers shouldn't
 # call close() from finish_response(), so if you close() anywhere but
 # the double-error branch here, you'll break asynchronous servers by
 # prematurely closing. Async servers must return from 'run()' without
 # closing if there might still be output to iterate over.
  ...
  self.setup_environ()
  self.result = application(self.environ, self.start_response)
  self.finish_response()
  ...

關鍵的3個步驟:

  1. setup_environ 繼續構建環境變量
  2. 接受application處理http請求的返回
  3. 完成http響應

setup_environ對env進行了進一步的包裝,附帶了請求的in/error,這樣讓使用env就可以對http請求進行讀寫。

def setup_environ(self):
 """Set up the environment for one request"""

 env = self.environ = self.os_environ.copy()
 self.add_cgi_vars() # 子類實現 self.environ.update(self.base_env)

 env['wsgi.input']  = self.get_stdin() # 注意沒有stdout
 env['wsgi.errors']  = self.get_stderr()
 env['wsgi.version']  = self.wsgi_version
 env['wsgi.run_once']  = self.wsgi_run_once
 env['wsgi.url_scheme'] = self.get_scheme()
 env['wsgi.multithread'] = self.wsgi_multithread
 env['wsgi.multiprocess'] = self.wsgi_multiprocess

 if self.wsgi_file_wrapper is not None:
  env['wsgi.file_wrapper'] = self.wsgi_file_wrapper

 if self.origin_server and self.server_software:
  env.setdefault('SERVER_SOFTWARE',self.server_software)

env的處理過程,可以理解成3步:1)附加server的運行信息 2)附加請求的http頭(協議信息) 3)附加請求的流信息。env,可以換個說法就是http請求的所有上下文環境。

application還接收一個回調函數start_response,主要是按照http協議的規范,生成響應狀態和response_header:

def start_response(self, status, headers,exc_info=None):
 """'start_response()' callable as specified by PEP 3333"""

 self.status = status
 self.headers = self.headers_class(headers)
 status = self._convert_string_type(status, "Status")
 assert len(status)>=4,"Status must be at least 4 characters"
 assert status[:3].isdigit(), "Status message must begin w/3-digit code"
 assert status[3]==" ", "Status message must have a space after code"

 return self.write

application對請求的處理:

def demo_app(environ,start_response):
 from io import StringIO
 stdout = StringIO()
 print("Hello world!", file=stdout)
 print(file=stdout)
 # http請求及環境
 h = sorted(environ.items())
 for k,v in h:
  print(k,'=',repr(v), file=stdout)
 # 回調寫入http_status, response_headers
 start_response("200 OK", [('Content-Type','text/plain; charset=utf-8')])
 # 返回處理結果response_body
 return [stdout.getvalue().encode("utf-8")]

響應仍然由ServerHandler寫入:

def finish_response(self):
 if not self.result_is_file() or not self.sendfile():
  for data in self.result:
   self.write(data)
  self.finish_content()

可以使用下面命令測試這個流程:

python -m wsgiref.simple_server
Serving HTTP on 0.0.0.0 port 8000 ...
127.0.0.1 - - [31/Jan/2021 21:43:05] "GET /xyz?abc HTTP/1.1" 200 3338

wsgi 小結

簡單小結wsgi的實現。在http請求的處理流程web-browser -> web-server -> wsgi -> web-application中,體現了分層的思想,每層做不同的事情:

  • web-server處理http/tcp協議,線程/進程的調度等底層實現
  • wsgi承上啟下,接受http請求,調用applicaiton處理請求,完成響應
  • application處理上層業務邏輯

小技巧

在wsgiref代碼中一樣有各種小的技巧, 學習后可以讓我們的代碼更pythonic。

環境變量都這樣設置:

def setup_environ(self):
 # Set up base environment
 env = self.base_environ = {}
 env['SERVER_NAME'] = self.server_name
 env['GATEWAY_INTERFACE'] = 'CGI/1.1'
 ...

我之前大概都是這樣寫:

def setup_environ(self):
 self.base_environ = {}
 self.base_environ['SERVER_NAME'] = self.server_name
 self.base_environ['GATEWAY_INTERFACE'] = 'CGI/1.1'

對比后,可以發現前面的寫法更簡潔一些。

比如流的持續寫入:

def _write(self,data):
 result = self.stdout.write(data)
 if result is None or result == len(data):
  return
 from warnings import warn
 warn("SimpleHandler.stdout.write() should not do partial writes",
  DeprecationWarning)
 while True:
  data = data[result:] # 持續的寫入,直到完成
  if not data:
   break
  result = self.stdout.write(data)

比如header的處理,實際上是把數組當作字典使用:

class Headers:
 """Manage a collection of HTTP response headers"""

 def __init__(self, headers=None):
  headers = headers if headers is not None else []
  self._headers = headers # 內部存儲使用數組
 
 def __setitem__(self, name, val):
  """Set the value of a header."""
  del self[name]
  self._headers.append(
   (self._convert_string_type(name), self._convert_string_type(val)))

 ....

 def __getitem__(self,name):
  """Get the first header value for 'name'

  Return None if the header is missing instead of raising an exception.

  Note that if the header appeared multiple times, the first exactly which
  occurrence gets returned is undefined. Use getall() to get all
  the values matching a header field name.
  """
  return self.get(name)

 def get(self,name,default=None):
  """Get the first header value for 'name', or return 'default'"""
  name = self._convert_string_type(name.lower())
  for k,v in self._headers:
   if k.lower()==name:
    return v
  return default

這樣對 Content-Type: application/javascript; charset=utf-8 這樣的值,可以使用下面方式使用:

if self.headers.get('content-type') is None:
 env['CONTENT_TYPE'] = self.headers.get_content_type()
else:
 env['CONTENT_TYPE'] = self.headers['content-type']

為什么用數組,而不是用字典呢?我猜測是因為header的特性是數據多為讀操作。

以上就是python wsgiref源碼解析的詳細內容,更多關于python wsgiref源碼的資料請關注腳本之家其它相關文章!

您可能感興趣的文章:
  • Python WSGI 規范簡介
  • 淺析Python 中的 WSGI 接口和 WSGI 服務的運行
  • Docker構建python Flask+ nginx+uwsgi容器
  • python 解決flask uwsgi 獲取不到全局變量的問題
  • python web框架 django wsgi原理解析
  • VPS CENTOS 上配置python,mysql,nginx,uwsgi,django的方法詳解
  • Python開發之Nginx+uWSGI+virtualenv多項目部署教程
  • CentOS7部署Flask(Apache、mod_wsgi、Python36、venv)
  • 詳解如何在Apache中運行Python WSGI應用
  • python 內置庫wsgiref的使用(WSGI基礎入門)

標簽:廊坊 臨汾 河池 重慶 長春 漢中 德宏 東莞

巨人網絡通訊聲明:本文標題《python wsgiref源碼解析》,本文關鍵詞  python,wsgiref,源碼,解析,python,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《python wsgiref源碼解析》相關的同類信息!
  • 本頁收集關于python wsgiref源碼解析的相關信息資訊供網民參考!
  • 推薦文章
    中文字幕日韩在线播放| 久久69精品久久久久久久电影好| 精品国产一区二区三区麻豆免费观看完整版 | 星空大象在线观看免费播放| 日韩欧美一二三区| 精品成人久久久| 亚洲黄色片网站| 日韩精品――中文字幕| 色女人综合av| 国产成人av在线播放| 亚洲欧洲无码一区二区三区| 男女羞羞免费视频| 欧美裸体男粗大视频在线观看| 国产在线观看免费视频软件| 日韩视频第一页| 日本中文字幕一区| 欧美亚洲综合色| 成人在线观看91| 久久丫精品久久丫| 夜夜爽夜夜爽精品视频| 欧美成欧美va| 国产精品99久久久久久久久 | 欧美成人福利视频| 日日夜夜精品视频免费| 日本熟女一区二区| www.亚洲自拍| 国产高清在线一区| 国产精品丝袜一区| jjzz黄色片| 亚洲一区www| 香蕉影院在线观看| 精品久久久三级| 日韩成人av一区二区| hitomi一区二区三区精品| 欧美综合国产精品久久丁香| 久久婷婷五月综合| 永久免费在线看片视频| 亚洲男女毛片无遮挡| 北条麻妃在线观看视频| 久久精品国产第一区二区三区最新章节| 日韩欧美国产系列| 亚洲综合在线第一页| 国产偷拍一区二区| 玖玖爱这里只有精品| 亚洲淫片在线视频| 蜜桃久久久久久久| 在线成人精品视频| 国产成人精品一区二区三区| 99视频在线观看免费| 成人淫片在线看| 欧美视频免费在线| 国产69精品久久久久毛片| 日本中文字幕高清| 欧美精品一本久久男人的天堂| 免费无码毛片一区二区app| 555www成人网| 高潮白浆女日韩av免费看| 国产精品一区二区三区99| 日韩在线视频第一页| 国产又粗又黄又爽| 日本在线播放视频| 久草免费新视频| 69sex久久精品国产麻豆| 精品久久久久久久中文字幕| 国产视频一二三四区| 香蕉在线观看视频| 欧美激情欧美狂野欧美精品| 一级片视频播放| 日本日本精品二区免费| 2019国产精品视频| 欧美另类久久久品| 99久久免费国产精精品| 一二三区视频在线观看| 中文字幕亚洲视频| 国产伦精品一区二区三区四区| 极品白嫩的小少妇| 91国内在线播放| 亚洲国产精品久久久久爰色欲| 亚洲精品久久区二区三区蜜桃臀| 麻豆av一区二区三区| 欧美一区二区三区婷婷月色| 日本亚洲一区二区| 日本精品一二三区| 国产美女被下药99| 精品国产三级电影在线观看| 中文字幕日韩三级| 欧美最顶级的aⅴ艳星| 日日夜夜精品免费| 一区二区三区四区在线视频| 亚洲精品美女在线观看| 91婷婷韩国欧美一区二区| 少妇按摩一区二区三区| 神马影院一区二区| 国产精品99一区| 国产精品美女www| 91精品久久久久久久久青青| 国产精品久久久久久久久久小说 | 蓝色福利精品导航| 国产一二三四在线| av网站在线观看不卡| 4438全国成人免费| 国产成人免费视频网站 | 国产精品午夜一区二区| 久久久亚洲综合网站| 亚洲精品国产视频| 无码人妻精品一区二区三区9厂| 亚洲午夜福利在线观看| 欧美中日韩免费视频| 久久男人资源视频| 日韩精品视频在线观看免费| 国内精品模特av私拍在线观看| 91久久国产精品91久久性色| 亚洲国产一区二区精品视频 | 日韩中文字幕在线精品| 色综合久久久久综合体桃花网| 国产伦精品一区二区三区免费| 国产高清第一页| 特色特色大片在线| 中文精品一区二区三区| 国内外成人免费激情在线视频 | 国产大片aaa| 欧美婷婷久久| 欧美日韩ab片| 亚洲色图激情小说| 亚洲天堂av在线免费| 亚洲麻豆国产自偷在线| 国产91在线|亚洲| 成人不卡免费av| 欧美午夜精品伦理| 久久久精品免费视频| 国产91精品久久久| 5g国产欧美日韩视频| 日韩中文字幕欧美| 亚洲福利精品在线| 亚洲成a人片综合在线| 成人av网站免费观看| 亚洲va男人天堂| 成人av在线播放网址| 中文字幕剧情在线观看| 国产精品久久久久久久美男 | 国产精品视频久| 久久久亚洲国产天美传媒修理工| 日韩av中文字幕在线| 色综合一个色综合| 欧美中文字幕不卡| 精品国产伦一区二区三区观看体验| 欧美极品美女电影一区| 日韩不卡av| 玖玖爱在线观看| 一级黄色短视频| 天堂在线视频免费| 国产成人手机在线| 日韩一卡二卡在线| 在线观看国产成人| 久草国产精品视频| 成人毛片一区二区三区| 特黄视频免费看| 九九精品视频在线| 天天免费综合色| 麻豆精品一区二区三区| 九九热最新地址| 亚洲系列第一页| 久久精品官网| 老司机久久99久久精品播放免费| 国产一区二区三区av电影| 国产午夜久久久久| 欧美日韩久久久久| 中文字幕综合一区| 日韩有码在线电影| 国产精品二区在线| 毛片毛片毛片毛片毛片毛片毛片毛片毛片 | 综合分类小说区另类春色亚洲小说欧美| 国产精品免费视频观看| 午夜一区在线观看| 波多野结衣啪啪| 精品人妻一区二区三区浪潮在线 | 99九九电视剧免费观看| 日韩一二区视频| 全黄一级裸体片| 国产美女在线精品免费观看| 永久免费看av| 免费网站在线观看黄| 久久精品第一页| www欧美成人18+| 欧美日韩免费观看一区三区| 国产精品青青在线观看爽香蕉| 91亚洲va在线va天堂va国| 欧洲另类一二三四区| 久久精品动漫| 亚洲大片一区二区三区| 91福利在线观看| 日韩一区二区三区电影在线观看 | 久久久一本精品99久久精品66| 国产主播欧美精品| 国产乱码精品一区二区三区卡| 一区二区不卡在线视频 午夜欧美不卡'| 91视频最新| 国产精品乱码| 免费亚洲一区二区| 中文字幕在线观看成人| 国产成人精品在线看| 亚洲成人午夜电影| 亚洲国产精品久久久久| 亚洲在线www| 俺去亚洲欧洲欧美日韩| 日韩午夜视频在线观看| 天堂av.com| 在线观看免费观看在线| 欧美日韩在线免费视频| 欧美xxxx黑人又粗又长精品| 夫妇露脸对白88av| 日韩国产欧美三级| 久久久久久免费| 最新热久久免费视频| 欧美大片在线看免费观看| 国产在线精品一区免费香蕉| 激情五月婷婷久久| 虎白女粉嫩尤物福利视频| 成人在线视频免费播放| 伊人网av在线| 日韩精品极品视频免费观看| 91免费在线视频网站| 国产美女网站在线观看| 国产精品suv一区二区| 99精品视频免费在线观看| 亚洲经典一区二区三区| 亚洲一二三区不卡| 成人在线中文字幕| 四虎1515hh.com| 中文字幕在线观看不卡视频| 久久久综合香蕉尹人综合网 | 夜夜躁日日躁狠狠久久88av| 又大又硬又爽免费视频| 免费中文字幕在线| 欧美aaa在线| 日韩亚洲欧美一区二区三区| 国产精品影院在线观看| 欧美一区在线直播| 国产伦精品一区二区三区高清| av网站在线不卡| 亚洲精品久久久蜜桃动漫| 亚洲天堂开心观看| 99精品视频网站| 中文字幕精品亚洲| 91香蕉视频污在线观看| 一区二区三区在线视频播放| 欧美男人的天堂一二区| 午夜精品久久久久久久99黑人| 黄色av免费在线播放| 国产综合色在线| 国产欧美中文字幕| 久久婷婷丁香| 国产精品日韩欧美| 在线观看毛片视频| 久久久免费观看视频| 午夜精品一区二区三区视频| 欧美激情综合网| 日韩精品极品毛片系列视频| 日韩欧美精品免费| 麻豆久久一区二区| 欧美成人午夜激情在线| 一级黄色片在线免费观看| 色噜噜日韩精品欧美一区二区| 精品一区二区免费在线观看| 日本最新高清不卡中文字幕| 亚洲精品午夜久久久久久久| 国产精品一级黄| 欧美高清视频免费观看| 欧美日韩中文字幕视频| 亚洲免费观看高清| 色狠狠av一区二区三区| 久久国产午夜精品理论片最新版本| 天堂网av在线播放| 国产精品免费久久久久久| 午夜在线视频观看| 欧美精品v日韩精品v国产精品| 国产婷婷一区二区| 久久av喷吹av高潮av| 久久精品国产精品青草| 国产精品va在线| 国产一级特黄a高潮片| 精品毛片乱码1区2区3区| 天天夜碰日日摸日日澡性色av| 亚州精品国产精品乱码不99按摩| 亚洲天堂色网站| 黄色一级片免费播放| 99精品久久久久久中文字幕| 欧美欧美欧美欧美首页| 日本熟妇人妻中出| 婷婷激情综合网| 亚洲一区日韩精品| 久久精品国产清高在天天线| 午夜精品一区二区三区视频免费看 | 欧美激情一区二区三区蜜桃视频| 成人激情视频在线播放| 婷婷在线免费观看| 亚洲一区尤物| 欧美日韩一二三区| 欧美一区二区三区久久久| 韩日午夜在线资源一区二区| 久久久夜色精品亚洲| 999香蕉视频| 色综合天天综合狠狠| 北条麻妃av高潮尖叫在线观看| 狠狠色综合播放一区二区| 中文字幕第4页| 蜜桃网站在线观看| 手机免费看av| 成人a免费在线看| 九九视频这里只有精品| 中文字幕第二区| 精品国产三级a在线观看| 日本丰满少妇裸体自慰| 在线观看av一区| aa免费在线观看| 国产精品色哟哟| 欧美日韩最好看的视频| 91在线第一页| 国产又粗又长又黄| 国产综合色香蕉精品| 中文字幕码精品视频网站| 亚洲专区在线视频| 国产精品久久久久久久久免费相片 | 亚洲精品不卡| 麻豆精品国产91久久久久久| 日韩av手机在线| 91成年人视频| 免费av在线一区| 日批在线观看视频| 午夜激情久久久| 国产一区二区播放| 亚洲free嫩bbb| 久草中文综合在线| 亚洲自拍偷拍二区| 亚洲日韩欧美一区二区在线| 国产吃瓜黑料一区二区| 国产很黄免费观看久久| 国产日本一区二区三区| 欧美极品美女视频| 超碰在线97免费| 日韩激情av在线免费观看| 看电视剧不卡顿的网站| 深田咏美中文字幕| 99超碰麻豆| 欧美mv日韩mv亚洲| 精品人妻av一区二区三区| 91大学生片黄在线观看| 亚洲国产日韩欧美在线动漫| 久热这里有精品| 95av在线视频| 日本欧美一区二区三区| 天天操天天爱天天爽| 欧美电影免费提供在线观看| 日本中文字幕免费观看| 成人免费大片黄在线播放| www.久久成人| 欧美日韩国产三区| 成人午夜碰碰视频| 久久久久久久久久久国产精品| 国产精品揄拍500视频| 成人免费毛片嘿嘿连载视频| 蜜臀av色欲a片无码精品一区| 日韩一级片网址| 日韩av在线发布| 中文字幕99| 91在线精品秘密一区二区| 久久福利小视频| 国产精品18久久久久久首页狼| 亚洲欧洲制服丝袜| 成人午夜精品福利免费| 中文字幕av一区二区三区人妻少妇| 国产91色在线| 制服丝袜成人动漫| 91网上在线视频| 亚洲 欧美 日韩 综合| 国产一线二线三线女| 久久精品国产v日韩v亚洲| 美国一区二区三区在线播放| 日本激情视频在线播放| 夜夜嗨av色一区二区不卡| 久久一区二区三区超碰国产精品| 爱情岛论坛亚洲首页入口章节| 伊人久久久久久久久久| 成人午夜视频在线观看| 亚洲av无码国产精品麻豆天美| 色偷偷9999www| 亚洲欧美一区二区三| 无码人妻丰满熟妇区毛片| 亚洲国产精品99久久| 成人激情小说网站| 波多野结衣一二区| 欧美性大战久久久久xxx| 日韩高清免费观看| 国产成人aaa| 日韩激情小视频| 缅甸午夜性猛交xxxx| 欧美日韩精品二区第二页| 三级黄色录像视频| 国产精品久久久一区二区三区| 欧美本精品男人aⅴ天堂| 男人的j进女人的j一区| 日韩国产亚洲欧美| 亚洲の无码国产の无码步美| 熟女视频一区二区三区| 国产精品专区一| 欧美亚洲另类制服自拍| 亚洲国产黄色片| 欧美亚洲国产一卡| 久久综合狠狠综合久久综合88 | 日本国产一区二区|