精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

復雜推理模型從服務器移植到Web瀏覽器的理論和實戰

開發 開發工具 服務器
隨著機器學習的應用面越來越廣,能在瀏覽器中跑模型推理的Javascript框架引擎也越來越多了。在項目中,前端同學可能會找到一些跑在服務端的python算法模型,很想將其直接集成到自己的代碼中,以Javascript語言在瀏覽器中運行。

一 背景

隨著機器學習的應用面越來越廣,能在瀏覽器中跑模型推理的Javascript框架引擎也越來越多了。在項目中,前端同學可能會找到一些跑在服務端的python算法模型,很想將其直接集成到自己的代碼中,以Javascript語言在瀏覽器中運行。

對于一部分簡單的模型,推理的前處理、后處理比較容易,不涉及復雜的科學計算,碰到這種模型,最多做個模型格式轉化,然后用推理框架直接跑就可以了,這種移植成本很低。

而很大一部分模型會涉及復雜的前處理、后處理,包括大量的矩陣運算、圖像處理等Python代碼。這種情況一般的思路就是用Javascript語言將Python代碼手工翻譯一遍,這么做的問題是費時費力還容易出錯。

Pyodide作為瀏覽器中的科學計算框架,很好的解決了這個問題:瀏覽器中運行原生的Python代碼進行前、后處理,大量numpy、scipy的矩陣、張量等計算無需翻譯為Javascript,為移植節省了很多工作。本文就基于pyodide框架,從理論和實戰兩個角度,幫助前端同學解決復雜模型的移植這一棘手問題。

二 原理篇

Pyodide是個可以在瀏覽器中跑的WebAssembly(wasm)應用。它基于CPython的源代碼進行了擴展,使用emscripten編譯成為wasm,同時也把一大堆科學計算相關的pypi包也編譯成了wasm,這樣就能在瀏覽器中解釋執行python語句進行科學計算了。所以pyodide也必然遵循wasm的各種約束。Pyodide在瀏覽器中的位置如下圖所示:??

??



1 wasm內存布局

這是wasm線性內存的布局:



?

??

??




Data數據段是從0x400開始的, Function Table表也在其中,起始地址為memoryBase(Emscripten中默認為1024,即0x400),STACKTOP為棧地址起始,堆地址起始為STACK_MAX。而我們實際更關心的是Javascript內存與wasm內存的互相訪問。

2 Javascript與Python的互訪

瀏覽器基于安全方面的考慮,防止wasm程序把瀏覽器搞崩潰,通過把wasm運行在一個沙箱化的執行環境中,禁止了wasm程序訪問Javascript內存,而Javascript代碼卻可以訪問wasm內存。因為wasm內存本質上是一個巨大的ArrayBuffer,接受Javascript的管理。我們稱之為“單向內存訪問”。

作為一個wasm格式的普通程序,pyodide被調用起來后,當然只能直接訪問wasm內存。



?

??

??




為了實現互訪,pyodide引入了proxy,類似于指針:在Javascript側,通過一個PyProxy對象來引用python內存里的對象;在Python側,通過一個JsProxy對象來引用Javascript內存里的對象。

在Javascript側生成一個PyProxy對象:

const arr_pyproxy = pyodide.globals.get('arr')  // arr是python里的一個全局對象

在Python側生成一個JsProxy對象:

import js 
from js import foo # foo是Javascript里的一個全局對象

互訪時的類型轉換分為如下三個等級:

  • 【自動轉換】對于簡單類型,如數字、字符串、布爾等,會被自動拷貝內存值,此時產生的就不是Proxy、而是最終的值了。
  • 【半自動轉換】非簡單的內置類型,都需要通過to_js()、to_py()方式來顯式轉換:
  • 對于Python內置的list、dict、numpy.ndarray等對象,不屬于簡單類型,不會自動轉換類型,必須通過pyodide.to_js()來轉,相應的會被轉成JS的list、map、TypedArray類型
  • 反過來也類似,通過to_py()方法,JS的TypedArray轉為memoryview,list、map轉為list、dict
  • 【手動轉換】各種class、function和用戶自定義類型,因為對方的語言沒有對應的現成類型,所以只能以proxy的形式存在,需要通過運算符來間接操縱,就像操縱提線木偶一樣。為了達到方便操縱的目的,pyodide對兩種語言進行了語法模擬,用一種語言里的操作符模擬另一種語言的類似行為。例如:JS中的let a=new XXX(),在Python中就變為a=XXX.new()。



?

??

??




這里列舉了一部分,詳情可以查文檔(見文章底部)。



?

??

??




Javascript的模塊也可以引入到Python中,這樣Python就能直接調用該模塊的接口和方法了。例如,pyodide沒有編譯opencv包,可以使用opencv.js:

import pyodide 
import js.cv as cv2
print(dir(cv2))

這對于pyodide缺失的pypi包是個很好的補充。

三 實踐篇

我們從一個空白頁面開始。使用瀏覽器打開測試頁面(測試頁面見文章底部)。

1 初始化python

為了方便觀察運行過程,使用動態的方式加載所需js和執行python代碼。打開瀏覽器控制臺,依次運行以下語句:

function loadJS( url, callback ){ 
var script = document.createElement('script'),
fn = callback || function(){};
script.type = 'text/javascript';
script.onload = function(){
fn();
};
script.src = url;
document.getElementsByTagName('head')[0].appendChild(script);
}
// 加載opencv
loadJS('https://test-bucket-duplicate.oss-cn-hangzhou.aliyuncs.com/public/opencv/opencv.js', function(){
console.log('js load ok');
});

// 加載推理引擎onnxruntime.js。當然也可以使用其他推理引擎
loadJS('https://test-bucket-duplicate.oss-cn-hangzhou.aliyuncs.com/public/onnxruntime/onnx.min.js', function(){
console.log('js load ok');
});

// 初始化python運行環境
loadJS('https://test-bucket-duplicate.oss-cn-hangzhou.aliyuncs.com/public/pyodide/0.18.0/pyodide.js', function(){
console.log('js load ok');
});
pyodide = await loadPyodide({ indexURL : "https://test-bucket-duplicate.oss-cn-hangzhou.aliyuncs.com/public/pyodide/0.18.0/"});
await pyodide.loadPackage(['micropip']);



?

??

??




至此,python和pip就安裝完畢了,都位于內存文件系統中。我們可以查看一下python被安裝到了哪里:



?

??

??




注意,這個文件系統是內存里虛擬出來的,刷新頁面就丟失了。不過由于瀏覽器本身有緩存,所以刷新頁面后從服務端再次加載pyodide的引導js和主體wasm還是比較快的,只要不清理瀏覽器緩存。

2 加載pypi包

在pyodide初始化完成后,python系統自帶的標準模塊可以直接import。第三方模塊需要用micropip.install()安裝:

  • pypi.org上的純python包可以用micropip.install() 直接安裝
  • 含有C語言擴展(編譯為動態鏈接庫)的wheel包,需要對照官方已編譯包的列表
  • 在列表中的直接用micropip.install()安裝
  • 不在這個列表里的,就需要自己手動編譯后發布到服務器后再用micropip.install()安裝。

下圖展示了業內常用的兩種編譯為wasm的方式。



?

??

??




自己編譯wasm package的方法可參考官方手冊,大致步驟就是pull官方的編譯基礎鏡像,把待編譯包的setup.cfg文件放到模塊目錄里,再加上些hack的語句和配置(如果有的話),然后指定目標進行編譯。編譯成功后部署時,需要注意2點:

  • 設置允許跨域
  • 對于wasm格式的文件請求,響應Header里應當帶上:"Content-type": "application/wasm"

下面是一個自建wasm服務器的nginx/openresty示例配置:

location ~ ^/wasm/ { 
add_header 'Access-Control-Allow-Origin' "*";
add_header 'Access-Control-Allow-Credentials' "true";
root /path/to/wasm_dir;
header_filter_by_lua '
uri = ngx.var.uri
if string.match(uri, ".js$") == nil then
ngx.header["Content-type"] = "application/wasm"
end
';
}

回到我們的推理實例, 現在用pip安裝模型推理所需的numpy和Pillow包并將其import:

await pyodide.runPythonAsync(` 
import micropip
micropip.install(["numpy", "Pillow"])
`);

await pyodide.runPythonAsync(`
import pyodide
import js.cv as cv2
import js.onnx as onnxruntime
import numpy as np
`);

這樣python所需的opencv、onnxruntime包就已全部導入了。

3 opencv的使用

一般python里的圖片數組都是從JS里傳過來的,這里我們模擬構造一張圖片,然后用opencv對其resize。上面提到過,pyodide官方的opencv還沒編譯出來。如果涉及到的opencv方法調用有其他pypi包的替代品,那是最好的:比如,cv.resize可以用Pillow庫的PIL.resize代替(注意Pillow的resize速度比opencv的resize要慢);cv.threshold可以用numpy.where代替。 否則只能調用opencv.js的能力了。為了演示pyodide語法,這里都從opencv.js庫里調用。

await pyodide.runPythonAsync(` 
# 構造一個1080p圖片
h,w = 1080,1920
img = np.arange(h * w * 3, dtype=np.uint8).reshape(h, w, 3)

# 使用cv2.resize將其縮小為1/10
# 原python代碼:small_img = cv2.resize(img, (h_small, w_small))
# 改成調用opencv.js:
h_small,w_small = 108, 192
mat = cv2.matFromArray(h, w, cv2.CV_8UC3, pyodide.to_js(img.reshape(h * w * 3)))
dst = cv2.Mat.new(h_small, w_small, cv2.CV_8UC3)
cv2.resize(mat, dst, cv2.Size.new(w_small, h_small), 0, 0, cv2.INTER_NEAREST)
small_img = np.asarray(dst.data.to_py()).reshape(h_small, w_small, 3)
`);

傳參原則:除了簡單的數字、字符串類型可以直接傳,其他類型都需要通過pyodide.to_js()轉換后再傳入。 返回值的獲取也類似,除了簡單的數字、字符串類型可以直接獲取,其他類型都需要通過xx.to_py()轉換后獲取結果。

接著對一個mask檢測其輪廓:

await pyodide.runPythonAsync(` 
# 使用cv2.findContours來檢測輪廓。假設mask為二維numpy數組,只有0、1兩個值
# 原python代碼:contours = cv2.findContours(mask, cv2.RETR_CCOMP,cv2.CHAIN_APPROX_NONE)
# 改成調用opencv.js:
contours_jsproxy = cv2.MatVector.new() # cv2.Mat數組,對應opencv.js中的 contours = new cv.MatVector()語句
hierarchy_jsproxy = cv2.Mat.new()
mat = cv2.matFromArray(mask.shape[0], mask.shape[1], cv2.CV_8UC1, pyodide.to_js(mask.reshape(mask.size)))
cv2.findContours(mat, pyodide.to_js(contours_jsproxy), pyodide.to_js(hierarchy_jsproxy), cv2.RETR_LIST, cv2.CHAIN_APPROX_SIMPLE)
# contours js格式轉python格式
contours = []
for i in range(contours_jsproxy.size()):
c_jsproxy = contours_jsproxy.get(i)
c = np.asarray(c_jsproxy.data32S.to_py()).reshape(c_jsproxy.rows, c_jsproxy.cols, 2)
contours.append(c)
`);

4 推理引擎的使用

最后,用onnx.js加載模型并進行推理,詳細語法可參考onnx.js官方文檔。其他js版的推理引擎也都可以參考各自的文檔。

await pyodide.runPythonAsync(` 
model_url="onnx模型的地址"
session = onnxruntime.InferenceSession.new()
session.loadModel(model_url)
session.run(......)
`);

通過以上的操作,我們確保了一切都在python語法范圍內進行,這樣修改原始的Python文件就比較容易了:把不支持的函數替換成我們自定義的調用js的方法;原Python里的推理替換成調用js版的推理引擎;最后在Javascript主程序框架里加少許調用Python的膠水代碼就完成了。

5 掛載持久存儲文件系統

有時我們需要對一些數據持久保存,可以利用pyodide提供的持久化文件系統(其實是emscripten提供的),見手冊(文章底部)。

// 創建掛載點 
pyodide.FS.mkdir('/mnt');
// 掛載文件系統
pyodide.FS.mount(pyodide.FS.filesystems.IDBFS, {}, '/mnt');
// 寫入一個文件
pyodide.FS.writeFile('/mnt/test.txt', 'hello world');
// 真正的保存文件到持久文件系統
pyodide.FS.syncfs(function (err) {
console.log(err);
});

這樣文件就持久保存了。即使當我們刷新頁面后,仍可以通過掛載該文件系統來讀出里面的內容:

// 創建掛載點 
pyodide.FS.mkdir('/mnt');
// 掛載文件系統
pyodide.FS.mount(pyodide.FS.filesystems.IDBFS, {}, '/mnt');
// 寫入一個文件
pyodide.FS.writeFile('/mnt/test.txt', 'hello world');
// 真正的保存文件到持久文件系統
pyodide.FS.syncfs(function (err) {
console.log(err);
});

運行結果如下:



?

??

??




當然,以上語句可以在python中以Proxy的語法方式運行。

持久文件系統有很多用處。例如,可以幫我們在多線程(webworker)之間共享大數據;可以把模型文件持久存儲到文件系統里,無需每次都通過網絡加載。

6 打wheel包

單Python文件無需打包,直接當成一個巨大的字符串,交給pyodide.runPythonAsync()運行就好了。當有多個Python文件時,我們可以把這些python文件打成普通wheel包,部署到webserver,然后可以用micropip直接安裝該wheel包:

micropip.install("https://foo.com/bar-1.2.3-xxx.whl") 
from bar import ...

注意,打wheel包需要有__init__.py文件,哪怕是個空文件。

四 存在的缺陷

目前pyodide有如下幾個缺陷:

  • Python運行環境加載和初始化時間有點兒長,視網絡情況,幾秒到幾十秒都有可能。
  • pypi包支持的不完整。雖然pypi.org上的純python包都可以直接使用,但涉及到C擴展寫的包,如果官方還沒編譯出來,那就需要自己動手編譯了。
  • 個別很常用的包,例如opencv,還沒成功編譯出來;模型推理框架一個都沒有。不過還好可以通過相應的JS庫來彌補。
  • 如果python中調用了js庫的話:
  • 可能會產生一定的內存拷貝開銷(從wasm內存到JS內存的來回拷貝)。尤其是大數組作為參數或返回值,在速度要求高的場合下,額外的內存拷貝開銷就不能忽視了。
  • python庫的方法接口可能跟其對應的js庫的接口參數、返回值格式不一致,有一定的適配工作量。

五 總結

盡管有上述種種缺陷,得益于代碼移植的高效率和邏輯上1:1復刻的高可靠性保障,我們還是可以把這種方法運用到多種業務場景里,為推動機器學習技術的應用添磚加瓦。

鏈接:

1、測試頁面:https://test-bucket-duplicate.oss-cn-hangzhou.aliyuncs.com/public/pyodide/test.html

2、文檔:https://pyodide.org/en/stable/usage/type-conversions.html

3、官方已編譯包的列表:https://github.com/pyodide/pyodide/tree/main/packages

4、手冊:https://emscripten.org/docs/api_reference/Filesystem-API.html

 

責任編輯:武曉燕 來源: 51CTO專欄
相關推薦

2014-05-05 09:56:12

Web瀏覽器遠程監控

2025-06-10 03:30:00

2018-08-30 08:47:59

瀏覽器web服務器通信

2025-03-05 00:22:00

2018-06-07 10:45:41

瀏覽器服務器響應

2020-03-09 08:24:06

TengineWeb代理服務器

2021-05-07 13:56:13

Linux器監視服務器

2019-02-26 16:20:52

FTP服務器

2012-08-14 17:07:13

2013-11-22 09:58:36

2017-02-08 08:46:39

瀏覽器服務端亂碼

2024-04-07 00:00:00

服務器瀏覽器AI

2010-04-05 21:57:14

Netscape瀏覽器

2025-11-11 04:15:00

FastMCPMCP服務器

2012-07-04 17:00:06

獵豹瀏覽瀏覽器

2012-02-23 10:02:08

Web服務器應用服務器

2025-11-13 08:00:00

大推理模型AI人工智能

2011-06-21 10:06:48

2015-06-12 10:03:05

QQ瀏覽器

2012-03-20 11:31:58

移動瀏覽器
點贊
收藏

51CTO技術棧公眾號

a在线视频观看| 久久精品视频va| 丰满的少妇愉情hd高清果冻传媒 | 欧美午夜一区二区三区| 亚洲黄色一区二区三区| 伊人网视频在线| 亚洲成人免费| 亚洲精品美女久久| 在线观看免费污视频| 超碰在线97国产| 亚洲国产精品v| 懂色中文一区二区三区在线视频| 欧美精品韩国精品| 欧美jizzhd精品欧美巨大免费| 亚洲精品一区久久久久久| 久久6免费视频| 欲香欲色天天天综合和网| 成人免费一区二区三区在线观看| 国产视色精品亚洲一区二区| 一区不卡在线观看| 久久国产99| 免费不卡欧美自拍视频| 国产美女永久免费无遮挡| 一区二区三区免费在线看| 在线看国产一区二区| 日本手机在线视频| 久久综合之合合综合久久| 久久免费精品国产久精品久久久久 | 欧美亚州在线观看| 亚洲AV无码一区二区三区性 | 欧美日韩一区二区三区| 亚洲欧美国产精品桃花| 欧美高清电影在线| 成人午夜免费视频| 91精品久久久久久久久中文字幕| 丁香六月婷婷综合| 亚洲激情自拍| 欧美高清视频在线| 欧美成人777| 日韩人体视频| 亚洲国产天堂久久国产91| 中文字幕久久久久久久| 爱情电影网av一区二区| 欧美日韩你懂得| 日本xxxxxxx免费视频| 川上优av中文字幕一区二区| 亚洲曰韩产成在线| 成人高清dvd| 国产人成网在线播放va免费| 国产精品美女久久久久aⅴ国产馆| 欧美在线视频二区| 深夜视频在线免费| av电影在线观看一区| 国产精品一区二区欧美黑人喷潮水| 精品人妻一区二区三区含羞草| 狠狠色丁香婷婷综合| 国产欧洲精品视频| 一级淫片免费看| 国产在线播放一区二区三区| 国产欧美婷婷中文| 91精品人妻一区二区三区果冻| 日本不卡不码高清免费观看| 国产区亚洲区欧美区| 一本色道久久综合精品婷婷| 国产一区不卡精品| 久久国内精品一国内精品| 欧美日精品一区视频| 欧美日韩精品一区| 人成免费电影一二三区在线观看| 99re这里只有精品首页| 久久青青草综合| 国产片在线观看| 国产精品天干天干在线综合| 在线不卡日本| av免费在线免费观看| 亚洲一区二区av在线| 91成人在线观看喷潮教学| 涩涩在线视频| 欧美午夜不卡在线观看免费| 在线a免费观看| 97久久超碰| 亚洲欧美国产精品va在线观看| 国产精品美女高潮无套| 91成人影院| 午夜精品www| 日韩精选在线观看| 国产在线精品视频| 国产三区二区一区久久| 成人在线观看一区| 亚洲免费观看视频| 无码播放一区二区三区| 国产精品原创视频| 精品国产91亚洲一区二区三区婷婷| 久久性爱视频网站| 欧美熟乱15p| 欧美激情极品视频| 国产精品熟女视频| 国产成人精品网址| 日本高清一区| 免费在线播放电影| 在线观看一区日韩| 色悠悠在线视频| 成人亚洲一区| 性金发美女69hd大尺寸| 影音先锋国产在线| 97精品久久久午夜一区二区三区| 亚洲国产欧美日韩| 日本а中文在线天堂| 欧美一区二区三区在线电影| 女人被狂躁c到高潮| 久久精品国内一区二区三区水蜜桃| 97久久精品视频| 国产又粗又猛又黄又爽无遮挡| 97aⅴ精品视频一二三区| 亚洲一区精彩视频| 裤袜国产欧美精品一区| 精品久久久久一区| 一本一本久久a久久| 亚洲自拍另类| 国产精品99久久久久久久| av在线播放av| 日韩欧美黄色动漫| 亚洲色偷偷色噜噜狠狠99网| 在线中文字幕第一区| 国产精品视频免费在线| 免费a在线观看| 亚洲aaa精品| 色欲无码人妻久久精品| 欧美丰满老妇| 国产精品久久久久77777| 午夜成人鲁丝片午夜精品| 一区二区三区日韩精品| 国产高清999| 欧美国产一级| 国产精品日韩专区| 国产福利在线视频| 色综合视频一区二区三区高清| 久草免费资源站| 欧美aⅴ99久久黑人专区| 51精品国产人成在线观看| 2017亚洲天堂1024| 欧美三片在线视频观看 | 成人激情视频网站| 91麻豆天美传媒在线| 91精品一久久香蕉国产线看观看| 一个人看的www久久| 中文字幕在线天堂| 久久免费午夜影院| 91热这里只有精品| 欧美日韩久久精品| 国产精品日韩电影| 免费看a在线观看| 欧美日韩二区三区| 日韩精品123区| 国产又黄又大久久| 日本一道在线观看| 天堂精品在线视频| 国内精品视频一区| 亚洲日本在线播放| 色哟哟精品一区| 亚洲精品视频网址| 激情深爱一区二区| 国产一区二区三区播放| 粉嫩一区二区三区四区公司1| 久久久久久免费精品| 男人天堂手机在线观看| 精品久久久一区二区| 亚洲a v网站| 蜜桃一区二区三区四区| 国产精品久久成人免费观看| 91成人入口| 欧美在线一级视频| jizz亚洲| 日韩欧美国产麻豆| 在线观看中文字幕视频| 国产农村妇女毛片精品久久麻豆 | 中文字幕制服丝袜成人av| 91丝袜超薄交口足| 亚洲国产清纯| 日韩电影免费观看高清完整| 青娱乐极品盛宴一区二区| 欧美国产极速在线| 清纯唯美亚洲色图| 在线不卡欧美精品一区二区三区| 青青草国产在线观看| 91丨porny丨国产入口| 三级视频中文字幕| 欧美激情 亚洲a∨综合| 久久国产精品久久精品国产| jizz欧美| 久久人人爽人人爽人人片av高请| 国产午夜在线视频| 欧美一级xxx| 国产免费一区二区三区四区五区| 成人欧美一区二区三区1314| 最近中文字幕无免费| 久久国产三级精品| 国产在线精品91| 欧美成人激情| 欧美亚洲爱爱另类综合| 高清在线一区二区| 日韩美女福利视频| 国产探花在线观看| 中文字幕日韩av电影| 色呦呦免费观看| 欧美人体做爰大胆视频| 久久国产视频精品| 中文字幕一区二区三| 亚洲成av人片一区二区| 国产人妻777人伦精品hd| 欧洲杯半决赛直播| 国产精品一区二区三区观看| 久久亚洲国产精品尤物| 68精品久久久久久欧美| 2024最新电影免费在线观看| 亚洲区中文字幕| 乱色精品无码一区二区国产盗| 欧美日韩久久一区| 日韩美一区二区| 亚洲不卡av一区二区三区| 特黄一区二区三区| 国产欧美一区二区三区鸳鸯浴| 国产免费a级片| 国产一区二区h| 午夜剧场高清版免费观看| 久久精品五月| 成人一对一视频| 欧美先锋影音| 日本一道在线观看| 999成人精品视频线3| 日本视频一区二区不卡| 色婷婷精品视频| 国产在线播放一区二区| 欧美日韩国产一区二区在线观看| 国产欧美日韩中文字幕在线| 精品日韩视频| 国产精品高清网站| 日韩网站中文字幕| 热久久免费视频精品| 欧产日产国产精品视频| 欧美高清视频在线观看| 欧美理论电影| 欧美国产亚洲视频| 日本h片在线观看| 欧美高跟鞋交xxxxhd| 免费看电影在线| 久久久久女教师免费一区| 欧美人与牲禽动交com| 欧美激情免费看| av免费不卡国产观看| 性欧美办公室18xxxxhd| 性xxxxfreexxxxx欧美丶| 91高清视频免费| 欧美精品日日操| 国产精品扒开腿做| 欧美成人黄色| 91香蕉国产在线观看| 欧美影院视频| 国产三级精品在线不卡| 亚洲精品合集| 亚洲欧美精品| 午夜精品999| 国产原创中文在线观看| 国产一区二区三区的电影| 日韩一级在线免费观看| 日产国产欧美视频一区精品| 天天色综合社区| 国产一区二区三区视频在线播放| 91人妻一区二区三区| 成人高清视频在线观看| 免费成人深夜夜行p站| 国产日产欧美一区二区三区| 青青草华人在线视频| 一区二区成人在线观看| 日韩人妻无码一区二区三区99| 日韩欧美中文字幕在线播放| 中文字幕制服诱惑| 欧美一区二区三区精品| 全部免费毛片在线播放一个| 亚洲三级av在线| 国产欧美黑人| 欧美一级淫片丝袜脚交| 狂野欧美性猛交xxxx| 国产精品国产三级国产专区53 | 尤物视频在线观看国产| 色视频成人在线观看免| xxx在线播放| 女人18毛片水真多18精品| 亚洲1区2区3区4区| 伊人中文字幕在线观看| 91精品免费观看| 婷婷五月综合久久中文字幕| 中文字幕久久精品| 日本资源在线| 国产成人精品综合久久久| 国产美女视频一区二区| 久久精品国产第一区二区三区最新章节 | 欧美女人交a| 国产自偷自偷免费一区| 成人性色生活片| 亚洲天堂岛国片| 一区二区欧美视频| 国产中文字幕视频| 日韩亚洲欧美在线| 国产高清免费av在线| 精品国产一区二区三区久久久蜜臀| 精品美女在线观看| 你懂得网站在线| 米奇精品一区二区三区在线观看| 亚洲天堂手机| 99久久伊人网影院| 欧美精品在线第一页| 天堂91在线| zzjj国产精品一区二区| а√在线天堂官网| 成人久久精品视频| 沈樵精品国产成av片| 国产精品日韩三级| 免费不卡在线观看| 色噜噜在线观看| 亚洲在线中文字幕| 国产精品探花视频| 国产亚洲精品久久久久久777| 国精产品一区一区三区mba下载| 国产精品女视频| 粉嫩一区二区三区四区公司1| 四虎影院一区二区| 日本午夜一区二区| 国产特级黄色录像| 欧美日韩另类字幕中文| 亚洲AV无码精品色毛片浪潮| 久久精品青青大伊人av| 99九九久久| 日韩欧美在线一区二区| 国产亚洲一级| 亚洲一区二区三区黄色| 亚洲黄色性网站| japanese国产| 欧美精品一区二区三区国产精品| 日本a人精品| 一区精品视频| 久久爱另类一区二区小说| 成人在线手机视频| 色8久久人人97超碰香蕉987| 你懂得在线网址| 国产精品久久久久久av| 欧美综合久久| 羞羞的视频在线| 国产精品毛片高清在线完整版| 中文字幕视频免费观看| 色阁综合伊人av| 日韩成人精品一区二区三区| 宅男av一区二区三区| 九九视频精品免费| 国产在线免费看| 欧美一级国产精品| 欧美人体视频xxxxx| 激情小说综合区| 老鸭窝毛片一区二区三区| 久久精品国产亚洲av久| 91久久香蕉国产日韩欧美9色| 国产主播福利在线| 国产日韩欧美91| 欧美福利电影在线观看| 欧美一级片在线免费观看| 亚洲sss视频在线视频| 日本私人网站在线观看| 国产精品久久久久秋霞鲁丝 | www深夜成人a√在线| 欧美一区二区三区免费视频| 国产丝袜在线观看视频| 蜜桃精品久久久久久久免费影院 | 欧美午夜精品久久久久久孕妇| 在线看的av网站| 成人h视频在线观看| 夜夜精品视频| 麻豆视频免费在线播放| 日韩欧美的一区二区| 妞干网免费在线视频| 亚洲日本精品| 国产成人精品亚洲午夜麻豆| 日本午夜视频在线观看| 日韩在线精品视频| 豆花视频一区二区| 国产福利影院在线观看| 一区二区三区在线不卡| 美女欧美视频在线观看免费 | 欧美日韩在线精品一区二区三区激情综 | 九九热在线免费观看| 色噜噜狠狠狠综合曰曰曰| a看欧美黄色女同性恋| 欧美国产日韩在线播放| 亚洲欧洲中文日韩久久av乱码| 五月婷中文字幕| 成人网欧美在线视频| 亚洲伦理精品| 老司机成人免费视频| 亚洲女人天堂视频| 日韩免费高清视频网站| 88av.com|