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

Spring Boot 接口數據加解密就該這樣設計~

人工智能 機器學習
考慮到時間緊迫性,可采用對稱性加密方式,服務需要對接安卓、IOS、H5三端,另外考慮到H5端存儲密鑰安全性相對來說會低一些,故分針對H5和安卓、IOS分配兩套密鑰;

今天這篇文章聊一聊接口安全問題,涉及到接口的加密、解密。

圖片

和產品、前端同學對外需求后,梳理了相關技術方案, 主要的需求點如下:

  • 盡量少改動,不影響之前的業務邏輯;
  • 考慮到時間緊迫性,可采用對稱性加密方式,服務需要對接安卓、IOS、H5三端,另外考慮到H5端存儲密鑰安全性相對來說會低一些,故分針對H5和安卓、IOS分配兩套密鑰;
  • 要兼容低版本的接口,后面新開發的接口可不用兼容;
  • 接口有GET和POST兩種接口,需要都要進行加解密;

需求解析:

  • 服務端、客戶端和H5統一攔截加解密,網上有成熟方案,也可以按其他服務中實現的加解密流程來搞;
  • 使用AES放松加密,考慮到H5端存儲密鑰安全性相對來說會低一些,故分針對H5和安卓、IOS分配兩套密鑰;
  • 本次涉及客戶端和服務端的整體改造,經討論,新接口統一加 /secret/ 前綴來區分

按本次需求來簡單還原問題,定義兩個對象,后面用得著,

用戶類:

@Data
public class User {
private Integer id;
private String name;
private UserType userType = UserType.COMMON;
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime registerTime;
}

用戶類型枚舉類:

@Getter
@JsonFormat(shape = JsonFormat.Shape.OBJECT)
public enum UserType {
VIP("VIP用戶"),
COMMON("普通用戶");
private String code;
private String type;

UserType(String type) {
this.code = name();
this.type = type;
}
}

構造一個簡單的用戶列表查詢示例:

@RestController
@RequestMapping(value = {"/user", "/secret/user"})
public class UserController {
@RequestMapping("/list")
ResponseEntity<List<User>> listUser() {
List<User> users = new ArrayList<>();
User u = new User();
u.setId(1);
u.setName("boyka");
u.setRegisterTime(LocalDateTime.now());
u.setUserType(UserType.COMMON);
users.add(u);
ResponseEntity<List<User>> response = new ResponseEntity<>();
response.setCode(200);
response.setData(users);
response.setMsg("用戶列表查詢成功");
return response;
}
}

調用:localhost:8080/user/list

查詢結果如下,沒毛病:

{
"code": 200,
"data": [{
"id": 1,
"name": "boyka",
"userType": {
"code": "COMMON",
"type": "普通用戶"
},
"registerTime": "2022-03-24 23:58:39"
}],
"msg": "用戶列表查詢成功"
}

目前主要是利用ControllerAdvice來對請求和響應體進行攔截,主要定義SecretRequestAdvice對請求進行加密和SecretResponseAdvice對響應進行加密(實際情況會稍微復雜一點,項目中又GET類型請求,自定義了一個Filter進行不同的請求解密處理)。

好了,網上的ControllerAdvice使用示例非常多,我這把兩個核心方法給大家展示看看,相信大佬們一看就曉得了,不需多言。上代碼:

SecretRequestAdvice請求解密:

@ControllerAdvice
@Order(Ordered.HIGHEST_PRECEDENCE)
@Slf4j
public class SecretRequestAdvice extends RequestBodyAdviceAdapter {
@Override
public boolean supports(MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass){
return true;
}

@Override
public HttpInputMessage beforeBodyRead(HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) throws IOException {
//如果支持加密消息,進行消息解密。
String httpBody;
if (Boolean.TRUE.equals(SecretFilter.secretThreadLocal.get())) {
httpBody = decryptBody(inputMessage);
} else {
httpBody = StreamUtils.copyToString(inputMessage.getBody(), Charset.defaultCharset());
}
//返回處理后的消息體給messageConvert
return new SecretHttpMessage(new ByteArrayInputStream(httpBody.getBytes()), inputMessage.getHeaders());
}

/**
* 解密消息體
*
* @param inputMessage 消息體
* @return 明文
*/
private String decryptBody(HttpInputMessage inputMessage) throws IOException {
InputStream encryptStream = inputMessage.getBody();
String requestBody = StreamUtils.copyToString(encryptStream, Charset.defaultCharset());
// 驗簽過程
HttpHeaders headers = inputMessage.getHeaders();
if (CollectionUtils.isEmpty(headers.get("clientType"))
|| CollectionUtils.isEmpty(headers.get("timestamp"))
|| CollectionUtils.isEmpty(headers.get("salt"))
|| CollectionUtils.isEmpty(headers.get("signature"))) {
throw new ResultException(SECRET_API_ERROR, "請求解密參數錯誤,clientType、timestamp、salt、signature等參數傳遞是否正確傳遞");
}

String timestamp = String.valueOf(Objects.requireNonNull(headers.get("timestamp")).get(0));
String salt = String.valueOf(Objects.requireNonNull(headers.get("salt")).get(0));
String signature = String.valueOf(Objects.requireNonNull(headers.get("signature")).get(0));
String privateKey = SecretFilter.clientPrivateKeyThreadLocal.get();
ReqSecret reqSecret = JSON.parseObject(requestBody, ReqSecret.class);
String data = reqSecret.getData();
String newSignature = "";
if (!StringUtils.isEmpty(privateKey)) {
newSignature = Md5Utils.genSignature(timestamp + salt + data + privateKey);
}
if (!newSignature.equals(signature)) {
// 驗簽失敗
throw new ResultException(SECRET_API_ERROR, "驗簽失敗,請確認加密方式是否正確");
}

try {
String decrypt = EncryptUtils.aesDecrypt(data, privateKey);
if (StringUtils.isEmpty(decrypt)) {
decrypt = "{}";
}
return decrypt;
} catch (Exception e) {
log.error("error: ", e);
}
throw new ResultException(SECRET_API_ERROR, "解密失敗");
}
}

SecretResponseAdvice響應加密:

@ControllerAdvice
public class SecretResponseAdvice implements ResponseBodyAdvice {
private Logger logger = LoggerFactory.getLogger(SecretResponseAdvice.class);

@Override
public boolean supports(MethodParameter methodParameter, Class aClass){
return true;
}

@Override
public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse){
// 判斷是否需要加密
Boolean respSecret = SecretFilter.secretThreadLocal.get();
String secretKey = SecretFilter.clientPrivateKeyThreadLocal.get();
// 清理本地緩存
SecretFilter.secretThreadLocal.remove();
SecretFilter.clientPrivateKeyThreadLocal.remove();
if (null != respSecret && respSecret) {
if (o instanceof ResponseBasic) {
// 外層加密級異常
if (SECRET_API_ERROR == ((ResponseBasic) o).getCode()) {
return SecretResponseBasic.fail(((ResponseBasic) o).getCode(), ((ResponseBasic) o).getData(), ((ResponseBasic) o).getMsg());
}
// 業務邏輯
try {
String data = EncryptUtils.aesEncrypt(JSON.toJSONString(o), secretKey);
// 增加簽名
long timestamp = System.currentTimeMillis() / 1000;
int salt = EncryptUtils.genSalt();
String dataNew = timestamp + "" + salt + "" + data + secretKey;
String newSignature = Md5Utils.genSignature(dataNew);
return SecretResponseBasic.success(data, timestamp, salt, newSignature);
} catch (Exception e) {
logger.error("beforeBodyWrite error:", e);
return SecretResponseBasic.fail(SECRET_API_ERROR, "", "服務端處理結果數據異常");
}
}
}
return o;
}
}

OK, 代碼Demo擼好了,試運行一波:

請求方法:
localhost:8080/secret/user/list

header:
Content-Type:application/json
signature:55efb04a83ca083dd1e6003cde127c45
timestamp:1648308048
salt:123456
clientType:ANDORID

body體:
// 原始請求體
{
"page": 1,
"size": 10
}
// 加密后的請求體
{
"data": "1ZBecdnDuMocxAiW9UtBrJzlvVbueP9K0MsIxQccmU3OPG92oRinVm0GxBwdlXXJ"
}

// 加密響應體:
{
"data": "fxHYvnIE54eAXDbErdrDryEsIYNvsOOkyEKYB1iBcre/QU1wMowHE2BNX/je6OP3NlsCtAeDqcp7J1N332el8q2FokixLvdxAPyW5Un9JiT0LQ3MB8p+nN23pTSIvh9VS92lCA8KULWg2nViSFL5X1VwKrF0K/dcVVZnpw5h227UywP6ezSHjHdA+Q0eKZFGTEv3IzNXWqq/otx5fl1gKQ==",
"code": 200,
"signature": "aa61f19da0eb5d99f13c145a40a7746b",
"msg": "",
"timestamp": 1648480034,
"salt": 632648
}

// 解密后的響應體:
{
"code": 200,
"data": [{
"id": 1,
"name": "boyka",
"registerTime": "2022-03-27T00:19:43.699",
"userType": "COMMON"
}],
"msg": "用戶列表查詢成功",
"salt": 0
}

OK,客戶端請求加密-》發起請求-》服務端解密-》業務處理-》服務端響應加密-》客戶端解密展示,看起來沒啥問題,實際是頭天下午花了2小時碰需求,差不多花1小時寫好demo測試,然后對所有接口統一進行了處理,整體一下午趕腳應該行了吧,告訴H5和安卓端同學明兒上午聯調(不小的大家到這個時候發現貓膩沒有,當時確實疏忽了,翻了大車......)

次日,安卓端反饋,你這個加解密有問題,解密后的數據格式和之前不一樣,仔細一看,擦,這個userType和registerTime是不對勁,開始思考:這個能是哪兒的問題呢?1s之后,初步定位,應該是響應體的JSON.toJSONString的問題:

String data = EncryptUtils.aesEncrypt(JSON.toJSONString(o)),

Debug斷點調試,果然,是JSON.toJSONString(o)這一步驟轉換出了問題,那JSON轉換時是不是有高級屬性可以配置生成想要的序列化格式呢?FastJson在序列化時提供重載方法,找到其中一個"SerializerFeature"參數可以琢磨一下,這個參數是可以對序列化進行配置的,它提供了很多配置類型,其中感覺這幾個比較沾邊:

WriteEnumUsingToString,
WriteEnumUsingName,
UseISO8601DateFormat

對枚舉類型來說,默認是使用的WriteEnumUsingName(枚舉的Name), 另一種WriteEnumUsingToString是重新toString方法,理論上可以轉換成想要的樣子,即這個樣子:

@Getter
@JsonFormat(shape = JsonFormat.Shape.OBJECT)
public enum UserType {
VIP("VIP用戶"),
COMMON("普通用戶");
private String code;
private String type;

UserType(String type) {
this.code = name();
this.type = type;
}

@Override
public String toString(){
return "{" +
"\"code\":\"" + name() + '\"' +
", \"type\":\"" + type + '\"' +
'}';
}
}

結果轉換出來的數據是字符串類型"{"code":"COMMON", "type":"普通用戶"}",這個方法好像行不通,還有什么好辦法呢?思前想后,看文章開始定義的User和UserType類,標記數據序列化格式@JsonFormat,再突然想起之前看到過的一些文章,SpringMVC底層默認是使用Jackson進行序列化的,那好了,就用Jacksong實施唄,將SecretResponseAdvice中的序列化方法替換一下:

String data = EncryptUtils.aesEncrypt(JSON.toJSONString(o), secretKey);
換為:
String data =EncryptUtils.aesEncrypt(new ObjectMapper().writeValueAsString(o), secretKey);

重新運行一波,走起:

{
"code": 200,
"data": [{
"id": 1,
"name": "boyka",
"userType": {
"code": "COMMON",
"type": "普通用戶"
},
"registerTime": {
"month": "MARCH",
"year": 2022,
"dayOfMonth": 29,
"dayOfWeek": "TUESDAY",
"dayOfYear": 88,
"monthValue": 3,
"hour": 22,
"minute": 30,
"nano": 453000000,
"second": 36,
"chronology": {
"id": "ISO",
"calendarType": "iso8601"
}
}
}],
"msg": "用戶列表查詢成功"
}

解密后的userType枚舉類型和非加密版本一樣了,舒服了,== 好像還不對,registerTime怎么變成這個樣子了?原本是"2022-03-24 23:58:39"這種格式的,網上有很多解決方案,不過用在我們目前這個需求里面,就是有損改裝了啊,不太可取,遂去Jackson官網上查找一下相關文檔,當然Jackson也提供了ObjectMapper的序列化配置,重新再初始化配置ObjectMpper對象:

String DATE_TIME_FORMATTER = "yyyy-MM-dd HH:mm:ss";
ObjectMapper objectMapper = new Jackson2ObjectMapperBuilder()
.findModulesViaServiceLoader(true)
.serializerByType(LocalDateTime.class, new LocalDateTimeSerializer(
DateTimeFormatter.ofPattern(DATE_TIME_FORMATTER)))
.deserializerByType(LocalDateTime.class, new LocalDateTimeDeserializer(
DateTimeFormatter.ofPattern(DATE_TIME_FORMATTER)))
.build();

轉換結果:

{
"code": 200,
"data": [{
"id": 1,
"name": "boyka",
"userType": {
"code": "COMMON",
"type": "普通用戶"
},
"registerTime": "2022-03-29 22:57:33"
}],
"msg": "用戶列表查詢成功"
}

OK,和非加密版的終于一致了,完了嗎?感覺還是可能存在些什么問題,首先業務代碼的時間序列化需求不一樣,有"yyyy-MM-dd hh:mm:ss"的,也有"yyyy-MM-dd"的,還可能其他配置思考不到位的,導致和之前非加密版返回數據不一致的問題,到時候聯調測出來了也麻煩,有沒有一勞永逸的辦法呢?哎,這個時候如果你看過 Spring 源碼的話,就應該知道spring框架自身是怎么序列化的,照著配置應該就行嘛,好像有點道理,我這里不從0開始分析源碼了。

跟著執行鏈路,找到具體的響應序列化,重點就是RequestResponseBodyMethodProcessor,

protected <T> void writeWithMessageConverters(@Nullable T value, MethodParameter returnType, ServletServerHttpRequest inputMessage, ServletServerHttpResponse outputMessage) throws IOException, HttpMediaTypeNotAcceptableException, HttpMessageNotWritableException {
// 獲取響應的攔截器鏈并執行beforeBodyWrite方法,也就是執行了我們自定義的SecretResponseAdvice中的beforeBodyWrite啦
body = this.getAdvice().beforeBodyWrite(body, returnType, selectedMediaType, converter.getClass(), inputMessage, outputMessage);
if (body != null) {
// 執行響應體序列化工作
if (genericConverter != null) {
genericConverter.write(body, (Type)targetType, selectedMediaType, outputMessage);
} else {
converter.write(body, selectedMediaType, outputMessage);
}
}

進而通過實例化的AbstractJackson2HttpMessageConverter對象找到執行序列化的核心方法

-> AbstractGenericHttpMessageConverter:

public final void write(T t, @Nullable Type type, @Nullable MediaType contentType, HttpOutputMessage outputMessage) throws IOException, HttpMessageNotWritableException {
...
this.writeInternal(t, type, outputMessage);
outputMessage.getBody().flush();

}
-> 找到Jackson序列化 AbstractJackson2HttpMessageConverter:
// 從spring容器中獲取并設置的ObjectMapper實例
protected ObjectMapper objectMapper;

protected void writeInternal(Object object, @Nullable Type type, HttpOutputMessage outputMessage) throws IOException, HttpMessageNotWritableException {
MediaType contentType = outputMessage.getHeaders().getContentType();
JsonEncoding encoding = this.getJsonEncoding(contentType);
JsonGenerator generator = this.objectMapper.getFactory().createGenerator(outputMessage.getBody(), encoding);

this.writePrefix(generator, object);
Object value = object;
Class<?> serializationView = null;
FilterProvider filters = null;
JavaType javaType = null;
if (object instanceof MappingJacksonValue) {
MappingJacksonValue container = (MappingJacksonValue)object;
value = container.getValue();
serializationView = container.getSerializationView();
filters = container.getFilters();
}

if (type != null && TypeUtils.isAssignable(type, value.getClass())) {
javaType = this.getJavaType(type, (Class)null);
}

ObjectWriter objectWriter = serializationView != null ? this.objectMapper.writerWithView(serializationView) : this.objectMapper.writer();
if (filters != null) {
objectWriter = objectWriter.with(filters);
}

if (javaType != null && javaType.isContainerType()) {
objectWriter = objectWriter.forType(javaType);
}

SerializationConfig config = objectWriter.getConfig();
if (contentType != null && contentType.isCompatibleWith(MediaType.TEXT_EVENT_STREAM) && config.isEnabled(SerializationFeature.INDENT_OUTPUT)) {
objectWriter = objectWriter.with(this.ssePrettyPrinter);
}
// 重點進行序列化
objectWriter.writeValue(generator, value);
this.writeSuffix(generator, object);
generator.flush();
}

那么,可以看出SpringMVC在進行響應序列化的時候是從容器中獲取的ObjectMapper實例對象,并會根據不同的默認配置條件進行序列化,那處理方法就簡單了,我也可以從Spring容器拿數據進行序列化啊。SecretResponseAdvice進行如下進一步改造:

@ControllerAdvice
public class SecretResponseAdvice implements ResponseBodyAdvice {

@Autowired
private ObjectMapper objectMapper;

@Override
public Object beforeBodyWrite(....){
.....
String dataStr =objectMapper.writeValueAsString(o);
String data = EncryptUtils.aesEncrypt(dataStr, secretKey);
.....
}
}

經測試,響應數據和非加密版萬全一致啦,還有GET部分的請求加密,以及后面加解密慘遭跨域問題,后面有空再和大家聊聊。

責任編輯:武曉燕 來源: 碼猿技術專欄
相關推薦

2022-07-27 08:49:34

接口加密解密

2024-09-27 15:24:15

Spring數據加解密

2022-09-15 08:41:16

數據異構分庫分表

2025-05-14 04:00:00

2022-12-05 09:08:12

微服務灰度發布

2023-06-27 08:58:03

2025-08-26 02:45:00

活動效果評估體系

2022-06-04 12:25:10

解密加密過濾器

2021-03-09 13:18:53

加密解密參數

2023-03-06 08:49:02

加密和解密SpringBoot

2023-07-05 13:58:10

權限模型設計模式

2022-09-14 18:23:01

工具加解密接口

2025-04-15 02:25:00

2022-07-18 08:39:18

ACL訪問控制機制

2011-08-01 14:36:06

加密RSA

2025-02-03 08:16:56

2017-12-07 10:25:55

LinuxGPG加密解密

2021-04-20 10:50:38

Spring Boot代碼Java

2022-11-18 18:36:24

2022-02-16 23:58:41

Spring過濾器驗證碼
點贊
收藏

51CTO技術棧公眾號

国产一区二区在线不卡| 精品无码国产污污污免费网站| 黄网站视频在线观看| 国产精品一区三区| 91a在线视频| 五月天免费网站| 超碰成人免费| 在线区一区二视频| 亚洲中文字幕无码一区二区三区| 日韩一级片免费| 美国十次了思思久久精品导航 | 99国产精品久久久久久久| 一本一道久久a久久精品逆3p | 亚洲人成无码网站久久99热国产| 九一在线视频| 成人一区二区三区在线观看| 国产成人一区二区三区| 精品一级少妇久久久久久久| 欧美色女视频| 亚洲国产精品大全| 97人人爽人人| av高清一区| 精品magnet| 日本中文字幕一级片| 国产精品影院在线| 99国产精品久久久久久久久久 | jazzjazz国产精品久久| 欧美色视频一区| 37pao成人国产永久免费视频| fc2ppv国产精品久久| 国产目拍亚洲精品99久久精品| 国产伦精品一区二区三区免| 91国产精品一区| 日日夜夜一区二区| 国产91精品高潮白浆喷水| 外国一级黄色片| 91精品国产91久久久久久密臀| 亚洲色图第三页| 五月婷婷综合在线观看| swag国产精品一区二区| 欧美一级在线观看| 不用播放器的免费av| 全球最大av网站久久| 色先锋资源久久综合| 少妇高潮喷水在线观看| tube8在线hd| 亚洲成人高清在线| www.亚洲成人网| 日本在线观看高清完整版| 最新国产精品久久精品| 中文字幕欧美人与畜| 成年人免费在线视频| 国产亚洲一二三区| 欧美日韩在线一二三| 日韩a在线观看| 久久久久综合网| 日韩国产欧美一区| 国产在线网站| 亚洲国产岛国毛片在线| 五月天亚洲综合情| 欧美尤物美女在线| 亚洲免费观看高清完整版在线| 可以在线看黄的网站| fc2ppv国产精品久久| 一区二区三区久久久| av网站手机在线观看| 毛片电影在线| 欧美最猛性xxxxx直播| 日本www高清视频| 日韩久久99| 日韩一区二区三区视频| www.欧美com| 五月综合久久| 中文字幕亚洲欧美日韩在线不卡| 国产探花在线视频| 欧美日韩网址| 日本精品一区二区三区在线| 中文字幕二区三区| 国产精品一区一区| 久久精品ww人人做人人爽| 久久久资源网| 日韩毛片视频在线看| 久久这里只有精品8| 女海盗2成人h版中文字幕| 日本高清不卡一区| 国产乱码一区二区三区四区| 福利欧美精品在线| 亚洲天堂网在线观看| 看免费黄色录像| 99热精品在线| 91精品久久久久久久久久| 亚洲av无码一区二区三区dv| 久久综合色婷婷| 日本三级福利片| 亚洲色图官网| 4438成人网| 短视频在线观看| 重囗味另类老妇506070| 5278欧美一区二区三区| 91精品中文字幕| 91最新地址在线播放| 亚洲欧美国产不卡| 91www在线| 欧美老年两性高潮| 国产在线观看无码免费视频| 91精品国产麻豆国产在线观看| 88xx成人精品| 精品久久久久久亚洲综合网站| 91免费在线看| www.99riav| 欧美黄页免费| 亚洲精品一区二三区不卡| 色欲人妻综合网| 丝袜亚洲精品中文字幕一区| 国产精品v欧美精品v日韩| 香蕉视频在线播放| 一本大道av一区二区在线播放| 先锋资源在线视频| 欧美gayvideo| 国产精品美女www爽爽爽视频| 欧美在线 | 亚洲| 亚洲色图欧美激情| 在线观看高清免费视频| 欧美久久精品| 久久噜噜噜精品国产亚洲综合| 国产精品久久久久久久成人午夜| 欧美激情一区二区| 免费高清在线观看免费| 国产精品一区二区中文字幕| 久久久999国产| 中文字幕免费高清在线观看| 2020日本不卡一区二区视频| 国产日本在线播放| 伊色综合久久之综合久久| 久久精品成人一区二区三区| 国产一级精品毛片| 国产午夜精品一区二区三区视频 | 色婷婷综合久久久中字幕精品久久| 精品精品欲导航| 欧美高清视频一区二区三区| 精品一区二区日韩| 杨幂一区欧美专区| 啪啪av大全导航福利综合导航| 亚洲一品av免费观看| 天堂网免费视频| 久久色中文字幕| 国产精品第12页| gogogo高清在线观看一区二区| 欧美在线视频观看| 天天av综合网| 色先锋资源久久综合| 最近中文字幕免费| 日本91福利区| av电影一区二区三区| 久久久久久久久成人| 欧美夫妻性生活xx| 女人18毛片一区二区三区| 亚洲成人综合网站| 亚洲狠狠婷婷综合久久久久图片| 免费精品视频| 日韩av图片| 曰本一区二区| 欧美激情亚洲自拍| 四虎永久在线观看| 日韩欧美中文免费| 国产欧美一区二区三区在线观看视频| 美国毛片一区二区| 欧美交换配乱吟粗大25p| 亚洲日本va中文字幕| 97精品视频在线| 天堂а在线中文在线无限看推荐| 日韩欧中文字幕| 羞羞在线观看视频| 成人福利视频在线看| 亚洲熟女乱色一区二区三区| 精品色999| 51国偷自产一区二区三区| 国产高清在线a视频大全| 精品视频在线播放| 一级片在线免费观看视频| 一区二区三区精品视频在线| 熟女俱乐部一区二区视频在线| 久久er精品视频| 欧美一级视频免费看| 精品产国自在拍| 91深夜福利视频| 色资源二区在线视频| 少妇精69xxtheporn| 亚洲成a人片77777精品| 欧美在线观看一区| 久草视频免费在线| 久久色中文字幕| 亚洲熟女乱综合一区二区| 国产精品日韩精品欧美精品| 这里只有精品66| 性人久久久久| 91九色露脸| 国产成人精品一区二三区在线观看 | 丁香花在线高清完整版视频| 亚洲最大在线视频| 午夜精品久久久久久久91蜜桃| 日韩欧美亚洲成人| 欧美极品视频在线观看| 亚洲国产电影在线观看| 免费的av网站| 国产精品一区二区久激情瑜伽| 日韩在线xxx| 亚洲国产激情| 400部精品国偷自产在线观看| 亚洲精品国模| 国产v亚洲v天堂无码| 久久精品资源| 日本在线观看天堂男亚洲 | 四虎成人精品永久免费av九九| 国产区日韩欧美| 精品视频国内| 成人国内精品久久久久一区| 成人影院av| 性欧美激情精品| 午夜av在线免费观看| 综合国产在线观看| 精品亚洲综合| 日韩精品在线观看网站| 亚洲黄色在线播放| 欧美精品v国产精品v日韩精品| 亚洲中文一区二区| 五月天久久比比资源色| 国产一级在线播放| 亚洲精品国产a久久久久久| 秋霞网一区二区三区| 久久久久久久久蜜桃| 在线免费观看a级片| 成人一二三区视频| 91视频免费入口| 国产麻豆成人精品| 天堂av手机在线| 久久国产乱子精品免费女| 久久精品网站视频| 久久久精品性| chinese少妇国语对白| 美女精品在线| 国产男女激情视频| 久久夜色精品| 欧美婷婷精品激情| 蜜桃视频在线一区| 潘金莲激情呻吟欲求不满视频| 美女mm1313爽爽久久久蜜臀| 不卡的在线视频| 精品一区二区三区免费观看 | 美女又爽又黄免费视频| 高潮白浆女日韩av免费看| 国产精品xxxx喷水欧美| 精品久久久久久中文字幕| 日韩网红少妇无码视频香港| 精品久久中文字幕| 国产99久久久| 欧美自拍偷拍午夜视频| 最近日韩免费视频| 欧美日韩激情一区二区| 国产一区二区小视频| 日韩三级高清在线| 特级丰满少妇一级aaaa爱毛片| 亚洲国产精品国自产拍av秋霞 | 国产 中文 字幕 日韩 在线| 爽爽窝窝午夜精品一区二区| 欧美精品中文字幕一区二区| 国产精品一在线观看| 亚州欧美一区三区三区在线| 久久资源中文字幕| 美女av免费观看| 国产欧美日韩综合一区在线播放| 国产精品免费成人| 理论电影国产精品| 最好看的中文字幕| www.av亚洲| 精品无码国产污污污免费网站| 国产精品久久久久久久久图文区| 性色av无码久久一区二区三区| 亚洲制服丝袜在线| 中文字幕av影院| 欧美日本一区二区| 黄频网站在线观看| 亚洲人成绝费网站色www| 麻豆tv免费在线观看| 久久久久久久久91| 日韩福利一区| 亚洲一区二区三区视频播放| 西瓜成人精品人成网站| 亚洲三区视频| 夜夜爽av福利精品导航| 国产一线二线三线在线观看| 国产精品资源在线看| theav精尽人亡av| 亚洲欧洲色图综合| 天天操天天干视频| 欧美精三区欧美精三区| 婷婷五月综合激情| 久久成人av网站| 欧美二三四区| 国产精品有限公司| 日韩视频在线观看| 国产精品丝袜久久久久久消防器材 | 欧美另类高清videos| 中文字幕在线免费观看视频| aa成人免费视频| 国产高清欧美| 男女曰b免费视频| 成人激情动漫在线观看| 你懂得在线观看| 91黄视频在线观看| 少妇一级淫片免费看| 久久亚洲成人精品| 日韩一级二级| 欧美日韩一区二区三区在线观看免| 一二三区不卡| a在线观看免费视频| 2023国产一二三区日本精品2022| 激情综合网五月天| 欧美精品v日韩精品v韩国精品v| 国产香蕉在线| 欧美有码在线观看| 国产伦精品一区二区三区在线播放| 制服诱惑一区| 蜜桃av一区二区在线观看| 欧美性xxxx图片| 香蕉乱码成人久久天堂爱免费| 国产三级精品在线观看| 日韩中文字幕精品| 欧美xnxx| 亚洲不卡一卡2卡三卡4卡5卡精品| 欧美黄色精品| 97超碰免费在线观看| 亚洲欧洲精品一区二区三区| 中文字幕人妻色偷偷久久| 亚洲欧美日韩精品久久亚洲区| 国产免费拔擦拔擦8x在线播放| 国产精品jizz视频| 黄色欧美日韩| 日本少妇xxxx软件| 亚洲综合999| 韩国中文字幕hd久久精品| 欧美激情视频免费观看| 婷婷视频一区二区三区| 波多野结衣与黑人| 国产二区国产一区在线观看| 99热精品免费| 精品久久久久久无| 黑人极品ⅴideos精品欧美棵| 国产精品xxxx| 99在线精品视频在线观看| 男女一区二区三区| 精品久久久久久电影| 日本在线一二三| 国产精品狠色婷| 色乱码一区二区三区网站| 国产性生活一级片| 亚洲欧美日韩系列| www.久久伊人| 国语对白做受69| 欧美日韩播放| 成人黄色一级大片| 亚洲在线观看免费| 色视频免费在线观看| 国产精欧美一区二区三区| 成人精品电影| 亚欧美一区二区三区| 一区二区三区免费看视频| 天天干,夜夜操| 国产精品久久久久久久久久ktv | 91亚洲精品久久久蜜桃网站| 黄瓜视频在线免费观看| 最近中文字幕日韩精品| 免费精品一区| 国产精品秘入口18禁麻豆免会员| 国产视频不卡一区| 国产男男gay体育生白袜| 久久91亚洲精品中文字幕奶水| 日韩av系列| 中文字幕永久有效| 亚洲国产欧美一区二区三区丁香婷 | 精品一区二区在线视频| 国产一级片免费观看| 亚洲男人的天堂在线播放| 99久久久国产| 欧美日韩亚洲一| 亚洲欧美一区二区视频| 免费国产羞羞网站视频| 国产精品久久二区| 欧美午夜久久| 卡一卡二卡三在线观看| 日韩美女在线视频| 视频在线日韩| www.好吊操| 国产精品蜜臀在线观看| 日本免费不卡视频| 91精品视频大全| 天堂蜜桃一区二区三区 | 欧美午夜视频在线观看| 黄色视屏免费在线观看| 久久天堂国产精品|