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

Sourcery 的 Swift Package 命令行插件

移動開發 iOS
當使用此新的 Camera service? 進行單元測試時,我們希望確保 AVCaptureSession? 沒有被真的創建。我們僅僅希望確認 camera service? 被測試系統(SUT)正確的調用了,而不是去測試 camera service 本身。

什么是Sourcery?

Sourcery 是當下最流行的 Swift 代碼生成工具之一。其背后使用了 SwiftSyntax[1],旨在通過自動生成樣板代碼來節省開發人員的時間。Sourcery 通過掃描一組輸入文件,然后借助模板的幫助,自動生成模板中定義的 Swift 代碼。

示例

考慮一個為攝像機會話服務提供公共 API 的協議:

protocol Camera {
func start()
func stop()
func capture(_ completion: @escaping (UIImage?) -> Void)
func rotate()
}

當使用此新的 Camera service 進行單元測試時,我們希望確保 AVCaptureSession 沒有被真的創建。我們僅僅希望確認 camera service 被測試系統(SUT)正確的調用了,而不是去測試 camera service 本身。

因此,創建一個協議的 mock 實現,使用空方法和一組變量來幫助我們進行單元測試,并斷言(asset)進行了正確的調用是有意義的。這是軟件開發中非常常見的一個場景,如果你曾維護過一個包含大量單元測試的大型代碼庫,這么做也可能有點乏味。

好吧~不用擔心!Sourcery 會幫助你!?? 它有一個叫做 AutoMockable[2] 的模板,此模板會為任意輸入文件中遵守 AutoMockable 協議的協議生成 mock 實現。

注意:在本文中,我擴展地使用了術語 Mock,因為它與 Sourcery 模板使用的術語一致。Mock 是一個相當重載的術語,但通常,如果我要創建一個 雙重測試[3],我會根據它的用途進一步指定類型的名稱(可能是 Spy 、 Fake 、 Stub 等)。如果您有興趣了解更多關于雙重測試的信息,馬丁·福勒(Martin Fowler)有一篇非常好的文章,可以解釋這些差異。

現在,我們讓 Camera 遵守 AutoMockable。該接口的唯一目的是充當 Sourcery 的目標,從中查找并生成代碼。

import UIKit

// Protocol to be matched
protocol AutoMockable {}

public protocol Camera: AutoMockable {
func start()
func stop()
func capture(_ completion: @escaping (UIImage?) -> Void)
func rotate()
}

此時,可以在上面的輸入文件上運行 Sourcery 命令,指定 AutoMockable 模板的路徑:

sourcery --sources Camera.swift --templates AutoMockable.stencil --output .

本文通過提供一個 .sourcery.yml 文件來配置 Sourcery 插件。如果提供了配置文件或 Sourcery 可以找到配置文件,則將忽略與其值沖突的所有命令行參數。如果您想了解有關配置文件的更多信息,Sourcery的 repo 中有一節[4]介紹了該主題。

命令執行完畢后,在輸出目錄下會生成一個 模板名 加 .generated.swift 為后綴的文件。在此例是 ./AutoMockable.generated.swift:

// Generated using Sourcery 1.8.2 — https://github.com/krzysztofzablocki/Sourcery
// DO NOT EDIT
// swiftlint:disable line_length
// swiftlint:disable variable_name

import Foundation
#if os(iOS) || os(tvOS) || os(watchOS)
import UIKit
#elseif os(OSX)
import AppKit
#endif

class CameraMock: Camera {

//MARK: - start

var startCallsCount = 0
var startCalled: Bool {
return startCallsCount > 0
}
var startClosure: (() -> Void)?

func start() {
startCallsCount += 1
startClosure?()
}

//MARK: - stop

var stopCallsCount = 0
var stopCalled: Bool {
return stopCallsCount > 0
}
var stopClosure: (() -> Void)?

func stop() {
stopCallsCount += 1
stopClosure?()
}

//MARK: - capture

var captureCallsCount = 0
var captureCalled: Bool {
return captureCallsCount > 0
}
var captureReceivedCompletion: ((UIImage?) -> Void)?
var captureReceivedInvocations: [((UIImage?) -> Void)] = []
var captureClosure: ((@escaping (UIImage?) -> Void) -> Void)?

func capture(_ completion: @escaping (UIImage?) -> Void) {
captureCallsCount += 1
captureReceivedCompletion = completion
captureReceivedInvocations.append(completion)
captureClosure?(completion)
}

//MARK: - rotate

var rotateCallsCount = 0
var rotateCalled: Bool {
return rotateCallsCount > 0
}
var rotateClosure: (() -> Void)?

func rotate() {
rotateCallsCount += 1
rotateClosure?()
}

}

上面的文件(AutoMockable.generated.swift)包含了你對mock的期望:使用空方法實現與目標協議的一致性,以及檢查是否調用了這些協議方法的一組變量。最棒的是… Sourcery 為您編寫了這一切!??

怎么運行 Sourcery?

怎么使用 Swift package 運行 Sourcery?

至此你可能在想如何以及怎樣在 Swift package 中運行 Sourcery。你可以手動執行,然后講文件拖到包中,或者從包目錄中的命令運行腳本。但是對于 Swift Package 有兩種內置方式運行可執行文件:

  1. 通過命令行插件,可根據用戶輸入任意運行
  2. 通過構建工具插件,該插件作為構建過程的一部分運行。

在本文中,我將介紹 Sourcery 命令行插件,但我已經在編寫第二部分,其中我將創建構建工具插件,這帶來了許多有趣的挑戰。

創建插件包

讓我們首先創建一個空包,并去掉測試和其他我們現在不需要的文件夾。然后我們可以創建一個新的插件 ??Target?? 并添加 Sourcery 的二進制文件作為其依賴項。

為了讓消費者使用這個插件,它還需要被定義為一個產品:

// swift-tools-version: 5.6
import PackageDescription

let package = Package(
name: "SourceryPlugins",
products: [
.plugin(name: "SourceryCommand", targets: ["SourceryCommand"])
],
targets: [
// 1
.plugin(
name: "SourceryCommand",
// 2
capability: .command(
intent: .custom(verb: "sourcery-code-generation", description: "Generates Swift files from a given set of inputs"),
// 3
permissions: [.writeToPackageDirectory(reason: "Need access to the package directory to generate files")]
),
dependencies: ["Sourcery"]
),
// 4
.binaryTarget(
name: "Sourcery",
path: "Sourcery.artifactbundle"
)
]
)

讓我們一步一步地仔細查看上面的代碼:

  1. 定義插件目標。
  2. 以custom 為意圖,定義了 .command 功能,因為沒有任何默認功能( documentationGeneration 和 sourceCodeFormatting)與該命令的用例匹配。給動詞一個合理的名稱很重要,因為這是從命令行調用插件的方式。
  3. 插件需要向用戶請求寫入包目錄的權限,因為生成的文件將被轉儲到該目錄。
  4. 為插件定義了一個二進制目標文件。這將允許插件通過其上下文訪問可執行文件。

我知道我并沒有詳細介紹上面的一些概念,但如果您想了解更多關于命令插件的信息,這里有一篇由 Tibor B?decs 寫的超級棒的文章?。如果你還想了解更多關于 Swift Packages 中二級制的目標(文件),我同樣有一篇??現今 Swift 包中的二進制目標??。

編寫插件

現在已經創建了包,是時候編寫一些代碼了!我們首先在 Plugins/SourceryCommand 下創建一個名為 SourceryCommand.swift 的文件,然后添加一個 CommandPlugin 協議的結構體,這將作為該插件的入口:

import PackagePlugin
import Foundation

@main
struct SourceryCommand: CommandPlugin {
func performCommand(context: PluginContext, arguments: [String]) async throws {

}
}

然后我們為命令編寫實現:

func performCommand(context: PluginContext, arguments: [String]) async throws {
// 1
let configFilePath = context.package.directory.appending(subpath: ".sourcery.yml").string
guard FileManager.default.fileExists(atPath: configFilePath) else {
Diagnostics.error("Could not find config at: \(configFilePath)")
return
}
//2
let sourceryExecutable = try context.tool(named: "sourcery")
let sourceryURL = URL(fileURLWithPath: sourceryExecutable.path.string)

// 3
let process = Process()
process.executableURL = sourceryURL

// 4
process.arguments = [
"--disableCache"
]
// 5
try process.run()
process.waitUntilExit()

// 6
let gracefulExit = process.terminationReason == .exit && process.terminationStatus == 0
if !gracefulExit {
Diagnostics.error("?? The plugin execution failed")
}
}

讓我們仔細看看上面的代碼:

  1. 首先.sourcery.yml 文件必須在包的根目錄,否則將報錯。這將使 Sourcery 神奇的工作,并使包可配置。
  2. 可執行文件路徑的 URL 是從命令的上下文中檢索的。
  3. 創建一個進程,并將 Sourcery 的可執行文件的 URL 設置為其可執行文件路徑。
  4. 這一步有點麻煩。Sourcery 使用緩存來減少后續運行的代碼生成時間,但問題是這些緩存是在包文件夾之外讀取和寫入的文件。插件的沙箱規則不允許這樣做,因此--disableCache 標志用于禁用此行為并允許命令運行。
  5. 進程同步運行并等待。
  6. 最后,檢查進程終止狀態和代碼,以確保進程已正常退出。在任何其他情況下,通過Diagnostics API 向用戶告知錯誤。

就這樣!現在讓我們使用它

使用(插件)包

考慮一個用戶正在使用插件,該插件將依賴項引入了他們的 Package.swift 文件:

// swift-tools-version: 5.6
// The swift-tools-version declares the minimum version of Swift required to build this package.
import PackageDescription
let package = Package(
name: "SourceryPluginSample",
products: [
// Products define the executables and libraries a package produces, and make them visible to other packages.
.library(
name: "SourceryPluginSample",
targets: ["SourceryPluginSample"]),
],
dependencies: [
.package(url: "https://github.com/pol-piella/sourcery-plugins.git", branch: "main")
],
targets: [
.target(
name: "SourceryPluginSample",
dependencies: [],
exclude: ["SourceryTemplates"]
),
]
)

 注意,與構建工具插件不同,命令插件不需要應用于任何目標,因為它們需要手動運行。

用戶只使用了上面的 AutoMockable 模板(可以在 Sources/SourceryPluginSample/SourceryTemplates 下找到),與本文前面顯示的示例相匹配:

protocol AutoMockable {}

protocol Camera: AutoMockable {
func start()
func stop()
func capture(_ completion: @escaping (UIImage?) -> Void)
func rotate()
}

根據插件的要求,用戶還提供了一個位于 SourceryPluginSample 目錄下的 .sourcery.yml 配置文件:

sources:
- Sources/SourceryPluginSample
templates:
- Sources/SourceryPluginSample/SourceryTemplates
output: Sources/SourceryPluginSample

運行命令

用戶已經設置好了,但是他們現在如何運行包??? 有兩種方法:

命令行

運行插件的一種方法是用命令行??梢酝ㄟ^從包目錄中運行 swift package plugin --list 來檢索特定包的可用插件列表。然后可以從列表中選擇一個包,并通過運行 swift package <command's verb> 來執行,在這個特殊的例子中,運行: swift package sourcery-code-generation。

注意,由于此包需要特殊權限,因此 --allow-writing-to-package-directory 必須與命令一起使用。

此時,你可能會想,為什么我要費心編寫一個插件,仍然必須從命令行運行,而我可以用一個簡單的腳本在幾行 bash 中完成相同的工作?好吧,讓我們來看看 Xcode 14 中會出現什么,你會明白為什么我會提倡編寫插件??。

Xcode

這是運行命令插件最令人興奮的方式,但不幸的是,它僅在 Xcode 14 中可用。因此,如果您需要運行命令,但尚未使用 Xcode 14,請參閱命令行部分。

如果你正好在使用 Xcode 14,你可以通過在文件資源管理器中右鍵單擊包,從列表中找到要執行的插件,然后單擊它來執行包的任何命令。

下一步

這是插件的初始實現。我將研究如何改進它,使它更加健壯。和往常一樣,我非常致力于公開構建,并使我的文章中的所有內容都開源,這樣任何人都可以提交問題或創建任何具有改進或修復的 PRs。這沒有什么不同??, 這是 公共倉庫的鏈接。

此外,如果您喜歡這篇文章,請關注即將到來的第二部分,其中我將制作一個 Sourcery 構建工具插件。我知道這聽起來不多,但這不是一項容易的任務!

參考資料

[1] SwiftSyntax: ??https://github.com/apple/swift-syntax。??

[2] AutoMockable: ??https://github.com/krzysztofzablocki/Sourcery/blob/master/Templates/Templates/AutoMockable.stencil。??

[3] 雙重測試: ??https://en.wikipedia.org/wiki/Test_double。??

[4] repo: ??https://github.com/krzysztofzablocki/Sourcery/blob/master/guides/Usage.md#configuration-file。??

責任編輯:姜華 來源: Swift社區
相關推薦

2016-03-28 10:00:09

Swift命令程序

2022-08-10 09:02:08

SwiftDecodable

2015-07-01 09:15:46

linuxQuora命令行

2023-08-01 13:31:18

模型Alpacaicuna

2020-12-11 06:44:16

命令行工具開發

2020-12-10 16:16:08

工具代碼開發

2011-01-18 19:11:26

Postfix命令行

2010-11-16 11:50:21

oracle命令行登錄

2012-02-08 16:37:36

ibmdw

2010-07-15 10:47:22

Perl命令行

2010-03-10 17:23:37

Python 命令行參

2009-12-24 14:51:39

Linux命令行

2020-12-06 08:00:46

scanimage命令行Linux

2023-06-09 07:45:29

Kuberneteskubectl

2010-07-20 14:18:22

Perl命令行

2010-07-15 09:37:47

Perl命令行

2019-07-23 13:45:38

LinuxFedora權限

2009-07-20 09:55:30

華為命令行解析華為認證

2010-08-20 10:05:23

用戶命令

2010-09-01 14:23:54

Linux命令行開發
點贊
收藏

51CTO技術棧公眾號

香蕉久久国产av一区二区| 日本一区二区在线免费观看| 国产高清一级毛片在线不卡| 久久在线91| 在线中文字幕日韩| 91日韩精品视频| 欧美aaaxxxx做受视频| 99国产精品视频免费观看| 日韩天堂在线视频| 日韩大尺度视频| 中文字幕一区久| 国产精品久久久久7777按摩| 91丝袜脚交足在线播放| 99久热在线精品996热是什么| 精品久久影视| 日韩精品在线一区二区| 日本精品一区二区三区四区| 欧美成年黄网站色视频| 精品一区二区三区欧美| 久久久噜噜噜久久| 天堂资源在线视频| 高清精品视频| 欧美日本一区二区| 黄色免费福利视频| 国产一区久久精品| 久久综合一区二区| 成人在线视频网址| 一区二区www| 免费久久99精品国产自在现线| 国产一区二区日韩精品欧美精品| 久久精品国亚洲| 男人天堂新网址| 成人福利在线| 不卡视频免费播放| 国产一区香蕉久久| 黑人一级大毛片| 欧美久久99| 中文字幕在线国产精品| 加勒比精品视频| 蜜桃在线一区| 在线免费观看视频一区| 日本一本中文字幕| 麻豆传媒在线观看| 日本一区二区三区高清不卡| 国产精品xxxx| aaa一区二区三区| 奇米精品一区二区三区在线观看| 国模吧一区二区三区| 亚洲欧洲日韩综合二区| 粉嫩精品久久99综合一区| 成人18夜夜网深夜福利网| 欧美另类z0zxhd电影| 妞干网在线免费视频| 岛国av在线网站| 亚洲欧美另类小说视频| 亚洲国产一区二区三区在线| 人人妻人人澡人人爽久久av| 久久91精品国产91久久小草| 国产成人精品免高潮在线观看| 久草免费新视频| 欧美三级特黄| 欧美日本亚洲视频| 九九视频免费观看| 欧美视频不卡| 97国产精品视频| 久艹在线观看视频| 91精品国产乱码久久久久久久| 中文字幕无线精品亚洲乱码一区| 娇妻被老王脔到高潮失禁视频| 最新国产精品视频| 亚洲午夜国产成人av电影男同| 久久久久亚洲av无码专区桃色| 久久悠悠精品综合网| 亚洲国产高潮在线观看| 极品粉嫩小仙女高潮喷水久久| 久久动漫网址| 亚洲欧美中文字幕| 国产免费嫩草影院| 三上亚洲一区二区| 久久av在线看| 日本少妇激情舌吻| 国产精品久久久久久久免费软件| 欧美一级在线播放| 一级黄色在线视频| 日韩高清不卡一区| 成人在线播放av| 精品久久久无码中文字幕| 国产成人av影院| 久久精品国产美女| 99中文字幕| 无码人妻aⅴ一区二区三区有奶水| 香蕉成人久久| 国产精品稀缺呦系列在线| 一区二区三区精| 国产成人精品免费视频网站| 国产精品视频在线免费观看 | 好男人在线视频www| 国产成人精品免费看| 精品婷婷色一区二区三区蜜桃| 黄色在线视频观看网站| 亚洲欧美自拍偷拍| 成人性免费视频| 国产精品xxx| 日韩欧美精品三级| 性猛交ⅹxxx富婆video| 91精品国产福利在线观看麻豆| 久久久久久久999| 99精品在线播放| 蓝色福利精品导航| 国产精品一区二区你懂得| 内射无码专区久久亚洲| 国产日韩精品视频一区| 麻豆传媒网站在线观看| 欧美大片免费观看网址| 欧美精品三级日韩久久| 成人欧美精品一区二区| 国产最新精品| 国产69精品久久久久9| 波多野结衣家庭主妇| 国产综合色视频| 久久国产精品高清| 超碰在线caoporn| 日韩欧美一区二区三区| 永久免费黄色片| 卡通动漫精品一区二区三区| 中文字幕日韩精品在线| 国产网站在线看| 蜜桃av噜噜一区| 精品乱子伦一区二区三区| 大片免费在线看视频| 色老头久久综合| 免费观看一区二区三区| 清纯唯美日韩| 国产mv免费观看入口亚洲| 激情文学亚洲色图| 亚洲欧美日本在线观看| 久久精品视频免费观看| 91亚洲精品国产| 伊人久久精品| 色诱女教师一区二区三区| 精品少妇theporn| 国内精品在线播放| 亚洲成人自拍视频| 日韩大尺度黄色| 亚洲精品电影在线观看| 国产女人被狂躁到高潮小说| 免费成人在线视频观看| 欧美三级华人主播| 色综合亚洲图丝熟| 亚洲精品一线二线三线| caoporn91| 国产乱一区二区| 欧美精品久久96人妻无码| 福利视频亚洲| 中文在线资源观看视频网站免费不卡| 青青操免费在线视频| 高清日韩电视剧大全免费| 黄色一级片国产| 试看120秒一区二区三区| 久久亚洲私人国产精品va| 91成品人影院| 亚洲视频你懂的| 国产又大又黄又粗的视频| 亚洲va久久| 欧洲亚洲妇女av| 日韩在线免费播放| 色悠悠久久综合| 色一情一交一乱一区二区三区| 午夜综合激情| 欧美主播一区二区三区美女 久久精品人 | 国产一区二区电影| 青青在线免费视频| 88久久精品| 午夜免费久久久久| 外国精品视频在线观看| 亚洲国产日产av| 美女搡bbb又爽又猛又黄www| 99在线精品视频在线观看| 国产欧美韩日| 欧美xxx视频| 一本色道久久综合狠狠躁篇的优点 | 欧美成人精品免费| 欧美成人基地| 国产97人人超碰caoprom| www黄在线观看| 5858s免费视频成人| 麻豆视频在线观看| 91网址在线看| 羞羞的视频在线| 亚洲视频电影在线| 国产精品一区二区三区精品| av影院在线免费观看| 亚洲视频免费一区| 国产精品一级视频| 亚洲福利电影网| 精品国产成人亚洲午夜福利| 久久激情五月激情| 亚洲熟妇无码av在线播放| 看全色黄大色大片免费久久久| 日本精品在线视频| 黄色精品免费看| 亚洲国产精品va在线看黑人动漫 | 魔女鞋交玉足榨精调教| 久久精品99国产精品| 日韩欧美国产综合在线| 成人精品影视| 国产亚洲精品久久飘花| 日韩av首页| 欧美精品电影免费在线观看| 久久电影中文字幕| 欧美一区二区三区系列电影| 国产微拍精品一区| 亚洲欧美日韩综合aⅴ视频| 国产精品无码毛片| 日本成人在线一区| 久久99中文字幕| 91中文字幕精品永久在线| 国产精品对白一区二区三区| 欧美日韩五码| 高清亚洲成在人网站天堂| 91精品专区| 日韩的一区二区| 国产毛片毛片毛片毛片| 亚洲v中文字幕| 麻豆视频在线免费看| 国产三区在线成人av| 亚洲成年人在线观看| 蜜桃视频一区二区三区| 亚欧无线一线二线三线区别| 欧美激情国产在线| 免费看成人av| 国产成人高清精品免费5388| 成人不卡免费av| 亚洲成人自拍| 国产欧美日韩精品一区二区三区| 91黄色国产视频| 国产一区二区三区四区五区3d | 久操视频在线| 一区二区亚洲欧洲国产日韩| 色欲av永久无码精品无码蜜桃 | 亚洲婷婷噜噜| 亚洲精品短视频| 亚州男人的天堂| 日韩av在线影院| 亚洲 小说区 图片区 都市| 亚洲激情第一页| 少妇人妻偷人精品一区二区| 精品国产乱码久久久久久免费 | 亚洲深夜福利视频| 黄色在线视频观看网站| 亚洲午夜久久久久久久| 国内在线免费高清视频| 在线视频中文亚洲| 视频一区二区三区不卡| 久久人人爽人人爽人人片亚洲 | 亚洲男人的天堂一区二区| 可以免费看av的网址| 1024成人网色www| 在线观看成人毛片| 亚洲成av人片www| 国偷自拍第113页| 色乱码一区二区三区88| 中文字幕丰满人伦在线| 在线成人av网站| 亚洲国产中文字幕在线| 精品少妇一区二区三区视频免付费 | 欧美黑人3p| 日韩成人三级| 国产精品一二三在线观看| 亚洲国产导航| 无码人妻丰满熟妇区毛片| 久久精品国产在热久久| 中文字幕人妻无码系列第三区| 成人黄色在线视频| 精品欧美一区二区久久久| 国产精品盗摄一区二区三区| 欧美特级一级片| 天天色天天操综合| 中文字幕日本人妻久久久免费 | 捆绑凌虐一区二区三区| 国产亚洲精久久久久久| 99精品中文字幕| 午夜精品久久久久久久久久久| 日韩欧美在线观看免费| 欧美日韩日日骚| 日本美女一级视频| 中文字幕亚洲欧美日韩2019| 青青草原av在线| 国产成人一区二区三区电影| 久久伊人影院| 欧美系列一区| 亚洲高清激情| 自拍偷拍一区二区三区四区| 99久久精品国产毛片| 欧美福利在线视频| 精品久久久久久| 国内精品久久久久久久久久久| 亚洲人成网站在线播| 新版中文在线官网| 国产精品国模在线| 精品国产18久久久久久洗澡| 香蕉久久夜色| 一本久道久久久| 久久久久久国产精品日本| 久久久久久久国产精品影院| 免费视频网站www| 欧美日韩另类国产亚洲欧美一级| 日本免费一区视频| 九九久久国产精品| 国产毛片精品久久| 欧美一区二区三区在线免费观看| 亚洲激情五月| 日韩肉感妇bbwbbwbbw| 91视频精品在这里| 国产亚洲精品码| 欧美美女激情18p| 国产精品天堂| 日韩av电影免费观看高清| 9l亚洲国产成人精品一区二三| 一区二区三区四区欧美| 久久黄色网页| 亚洲一区二区乱码| 天天综合日日夜夜精品| 国产香蕉在线观看| 精品中文字幕在线观看| 九七影院97影院理论片久久| 精品一区在线播放| 国产欧美日韩一级| av免费观看不卡| 亚洲国产中文字幕在线视频综合| 国产伦精品一区二区三区免.费| 国产午夜一区二区| 深夜视频一区二区| 欧美在线一二三区| 日日嗨av一区二区三区四区| 亚洲成人av免费在线观看| 亚洲二区在线观看| 色综合免费视频| 97精品久久久中文字幕免费| 伊人久久噜噜噜躁狠狠躁| 日本精品福利视频| 国产精品888| 久久婷婷国产麻豆91| 日韩一级欧美一级| 波多野结衣精品| 国产一区不卡在线观看| 99国产精品视频免费观看一公开| 野战少妇38p| 婷婷中文字幕综合| 蜜桃视频在线免费| 国产精品99蜜臀久久不卡二区| 欧美综合视频| 热久久久久久久久| 樱桃国产成人精品视频| 蜜桃视频久久一区免费观看入口| 久久久久久九九九| 亚洲欧洲美洲国产香蕉| 爆乳熟妇一区二区三区霸乳| 欧美国产亚洲另类动漫| 一炮成瘾1v1高h| 欧美激情视频给我| 久久精品亚洲成在人线av网址| 91九色在线观看视频| 国产亚洲欧美一级| 国产又大又黄又爽| 欧美精品国产精品日韩精品| 女仆av观看一区| 91最新在线观看| 综合自拍亚洲综合图不卡区| 国内精品久久久久久久久久久 | 大桥未久一区二区三区| 成人av在线网| 乱子伦一区二区三区| www.xxxx精品| 玖玖玖免费嫩草在线影院一区| 国产精品wwwww| 亚洲老妇xxxxxx| 青青草av免费在线观看| 国产日韩欧美91| 亚洲美女色禁图| 日韩精品电影一区二区三区| 91精品国产91久久综合桃花| а√天堂8资源在线| 五月天亚洲综合情| 国产盗摄女厕一区二区三区| 日韩免费一级片| 日韩中文在线不卡| 久久久久高潮毛片免费全部播放| 欧美成人黑人猛交| 亚洲一区二区欧美| 风间由美一区| 豆国产97在线| 日韩精品一二三| 国产 日韩 欧美 成人| 在线视频一区二区| 欧美美女啪啪| 人妻巨大乳一二三区| 在线视频综合导航| heyzo高清中文字幕在线| 午夜视频久久久|