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

SpringBoot 加密解密新招

開發 前端
繼承HttpServletRequestWrapper,將請求中的流copy一份,復寫getInputStream和getReader方法供外部使用。每次調用后的getInputStream方法都是從復制出來的二進制數組中進行獲取,這個二進制數組在對象存在期間一致存在。

1. 介紹

在我們日常的Java開發中,免不了和其他系統的業務交互,或者微服務之間的接口調用

如果我們想保證數據傳輸的安全,對接口出參加密,入參解密。

但是不想寫重復代碼,我們可以提供一個通用starter,提供通用加密解密功能

2. 前置知識

2.1 hutool-crypto加密解密工具

hutool-crypto提供了很多加密解密工具,包括對稱加密,非對稱加密,摘要加密等等,這不做詳細介紹。

2.2 request流只能讀取一次的問題

2.2.1 問題:

在接口調用鏈中,request的請求流只能調用一次,處理之后,如果之后還需要用到請求流獲取數據,就會發現數據為空。

比如使用了filter或者aop在接口處理之前,獲取了request中的數據,對參數進行了校驗,那么之后就不能在獲取request請求流了

2.2.2 解決辦法

繼承HttpServletRequestWrapper,將請求中的流copy一份,復寫getInputStream和getReader方法供外部使用。每次調用后的getInputStream方法都是從復制出來的二進制數組中進行獲取,這個二進制數組在對象存在期間一致存在。

使用Filter過濾器,在一開始,替換request為自己定義的可以多次讀取流的request。

這樣就實現了流的重復獲取InputStreamHttpServletRequestWrapper。

import org.apache.commons.io.IOUtils;

import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStreamReader;

/**
 * 請求流支持多次獲取
 */
public class InputStreamHttpServletRequestWrapper extends HttpServletRequestWrapper {

    /**
     * 用于緩存輸入流
     */
    private ByteArrayOutputStream cachedBytes;

    public InputStreamHttpServletRequestWrapper(HttpServletRequest request) {
        super(request);
    }

    @Override
    public ServletInputStream getInputStream() throws IOException {
        if (cachedBytes == null) {
            // 首次獲取流時,將流放入 緩存輸入流 中
            cacheInputStream();
        }

        // 從 緩存輸入流 中獲取流并返回
        return new CachedServletInputStream(cachedBytes.toByteArray());
    }

    @Override
    public BufferedReader getReader() throws IOException {
        return new BufferedReader(new InputStreamReader(getInputStream()));
    }

    /**
     * 首次獲取流時,將流放入 緩存輸入流 中
     */
    private void cacheInputStream() throws IOException {
        // 緩存輸入流以便多次讀取。為了方便, 我使用 org.apache.commons IOUtils
        cachedBytes = new ByteArrayOutputStream();
        IOUtils.copy(super.getInputStream(), cachedBytes);
    }

    /**
     * 讀取緩存的請求正文的輸入流
     * <p>
     * 用于根據 緩存輸入流 創建一個可返回的
     */
    public static class CachedServletInputStream extends ServletInputStream {

        private final ByteArrayInputStream input;

        public CachedServletInputStream(byte[] buf) {
            // 從緩存的請求正文創建一個新的輸入流
            input = new ByteArrayInputStream(buf);
        }

        @Override
        public boolean isFinished() {
            return false;
        }

        @Override
        public boolean isReady() {
            return false;
        }

        @Override
        public void setReadListener(ReadListener listener) {

        }

        @Override
        public int read() throws IOException {
            return input.read();
        }
    }

}

HttpServletRequestInputStreamFilter

import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;

import static org.springframework.core.Ordered.HIGHEST_PRECEDENCE;

/**
 * @description:
 *      請求流轉換為多次讀取的請求流 過濾器
 */
@Component
@Order(HIGHEST_PRECEDENCE + 1)  // 優先級最高
public class HttpServletRequestInputStreamFilter implements Filter {

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {

        // 轉換為可以多次獲取流的request
        HttpServletRequest httpServletRequest = (HttpServletRequest) request;
        InputStreamHttpServletRequestWrapper inputStreamHttpServletRequestWrapper = new InputStreamHttpServletRequestWrapper(httpServletRequest);

        // 放行
        chain.doFilter(inputStreamHttpServletRequestWrapper, response);
    }
}

2.3 SpringBoot的參數校驗validation

為了減少接口中,業務代碼之前的大量冗余的參數校驗代碼

SpringBoot-validation提供了優雅的參數校驗,入參都是實體類,在實體類字段上加上對應注解,就可以在進入方法之前,進行參數校驗,如果參數錯誤,會拋出錯誤BindException,是不會進入方法的。

這種方法,必須要求在接口參數上加注解@Validated或者是@Valid,但是很多情況下,我們希望在代碼中調用某個實體類的校驗功能,所以需要如下工具類:

ParamException

import lombok.Getter;

import java.util.List;

/**
 * @description 自定義參數異常
 */
@Getter
public class ParamException extends Exception {

    private final List<String> fieldList;
    private final List<String> msgList;

    public ParamException(List<String> fieldList, List<String> msgList) {
        this.fieldList = fieldList;
        this.msgList = msgList;
    }
}

ValidationUtils

import javax.validation.ConstraintViolation;
import javax.validation.Validation;
import javax.validation.Validator;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;

/**
 * @description 驗證工具類
 */
public class ValidationUtils {

    private static final Validator VALIDATOR = Validation.buildDefaultValidatorFactory().getValidator();

    /**
     * 驗證數據
     * @param object 數據
     */
    public static void validate(Object object) throws CustomizeException {

        Set<ConstraintViolation<Object>> validate = VALIDATOR.validate(object);

        // 驗證結果異常
        throwParamException(validate);
    }

    /**
     * 驗證數據(分組)
     * @param object 數據
     * @param groups 所在組
     */
    public static void validate(Object object, Class<?> ... groups) throws CustomizeException {

        Set<ConstraintViolation<Object>> validate = VALIDATOR.validate(object, groups);

        // 驗證結果異常
        throwParamException(validate);
    }

    /**
     * 驗證數據中的某個字段(分組)
     * @param object 數據
     * @param propertyName 字段名稱
     */
    public static void validate(Object object, String propertyName) throws CustomizeException {
        Set<ConstraintViolation<Object>> validate = VALIDATOR.validateProperty(object, propertyName);

        // 驗證結果異常
        throwParamException(validate);

    }

    /**
     * 驗證數據中的某個字段(分組)
     * @param object 數據
     * @param propertyName 字段名稱
     * @param groups 所在組
     */
    public static void validate(Object object, String propertyName, Class<?> ... groups) throws CustomizeException {

        Set<ConstraintViolation<Object>> validate = VALIDATOR.validateProperty(object, propertyName, groups);

        // 驗證結果異常
        throwParamException(validate);

    }

    /**
     * 驗證結果異常
     * @param validate 驗證結果
     */
    private static void throwParamException(Set<ConstraintViolation<Object>> validate) throws CustomizeException {
        if (validate.size() > 0) {
            List<String> fieldList = new LinkedList<>();
            List<String> msgList = new LinkedList<>();
            for (ConstraintViolation<Object> next : validate) {
                fieldList.add(next.getPropertyPath().toString());
                msgList.add(next.getMessage());
            }

            throw new ParamException(fieldList, msgList);
        }
    }

}

2.4 自定義starter

自定義starter步驟:

  • 創建工廠,編寫功能代碼。
  • 聲明自動配置類,把需要對外提供的對象創建好,通過配置類統一向外暴露。
  • 在resource目錄下準備一個名為spring/spring.factories的文件,以org.springframework.boot.autoconfigure.EnableAutoConfiguration為key,自動配置類為value列表,進行注冊。

2.5 RequestBodyAdvice和ResponseBodyAdvice

  • RequestBodyAdvice是對請求的json串進行處理, 一般使用環境是處理接口參數的自動解密。
  • ResponseBodyAdvice是對請求相應的json傳進行處理,一般用于相應結果的加密。

3. 功能介紹

接口相應數據的時候,返回的是加密之后的數據接口入參的時候,接收的是解密之后的數據,但是在進入接口之前,會自動解密,取得對應的數據

4. 功能細節

加密解密使用對稱加密的AES算法,使用hutool-crypto模塊進行實現

所有的實體類提取一個公共父類,包含屬性時間戳,用于加密數據返回之后的實效性,如果超過60分鐘,那么其他接口將不進行處理。

如果接口加了加密注解EncryptionAnnotation,并且返回統一的json數據Result類,則自動對數據進行加密。如果是繼承了統一父類RequestBase的數據,自動注入時間戳,確保數據的時效性

如果接口加了解密注解DecryptionAnnotation,并且參數使用RequestBody注解標注,傳入json使用統一格式RequestData類,并且內容是繼承了包含時間長的父類RequestBase,則自動解密,并且轉為對應的數據類型

功能提供Springboot的starter,實現開箱即用

5. 代碼實現

https://gitee.com/springboot-hlh/spring-boot-csdn/tree/master/09-spring-boot-interface-crypto

5.1 項目結構

圖片圖片

5.2 crypto-common

5.2.1 結構

圖片圖片

5.3 crypto-spring-boot-starter

5.3.1 接口

圖片圖片

5.3.2 重要代碼

crypto.properties AES需要的參數配置;

# 模式    
crypto.mode=CTS
# 補碼方式 
crypto.padding=PKCS5Padding
# 秘鑰
crypto.key=testkey123456789
# 鹽
crypto.iv=testiv1234567890

spring.factories 自動配置文件;

org.springframework.boot.autoconfigure.EnableAutoCnotallow=\
        xyz.hlh.crypto.config.AppConfig

CryptConfig AES需要的配置參數;

import cn.hutool.crypto.Mode;
import cn.hutool.crypto.Padding;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.Getter;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;

import java.io.Serializable;

/**
 * @description: AES需要的配置參數
 */
@Configuration
@ConfigurationProperties(prefix = "crypto")
@PropertySource("classpath:crypto.properties")
@Data
@EqualsAndHashCode
@Getter
public class CryptConfig implements Serializable {

    private Mode mode;
    private Padding padding;
    private String key;
    private String iv;

}

AppConfig 自動配置類;

import cn.hutool.crypto.symmetric.AES;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import javax.annotation.Resource;
import java.nio.charset.StandardCharsets;

/**
 * @description: 自動配置類
 */
@Configuration
public class AppConfig {

    @Resource
    private CryptConfig cryptConfig;

    @Bean
    public AES aes() {
        return new AES(cryptConfig.getMode(), cryptConfig.getPadding(), cryptConfig.getKey().getBytes(StandardCharsets.UTF_8), cryptConfig.getIv().getBytes(StandardCharsets.UTF_8));
    }

}

DecryptRequestBodyAdvice 請求自動解密;

import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.SneakyThrows;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.MethodParameter;
import org.springframework.http.HttpInputMessage;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.context.request.RequestAttributes;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import org.springframework.web.servlet.mvc.method.annotation.RequestBodyAdvice;
import xyz.hlh.crypto.annotation.DecryptionAnnotation;
import xyz.hlh.crypto.common.exception.ParamException;
import xyz.hlh.crypto.constant.CryptoConstant;
import xyz.hlh.crypto.entity.RequestBase;
import xyz.hlh.crypto.entity.RequestData;
import xyz.hlh.crypto.util.AESUtil;

import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
import java.lang.reflect.Type;

/**
 * @description: requestBody 自動解密
 */
@ControllerAdvice
public class DecryptRequestBodyAdvice implements RequestBodyAdvice {

    @Autowired
    private ObjectMapper objectMapper;

    /**
     * 方法上有DecryptionAnnotation注解的,進入此攔截器
     * @param methodParameter 方法參數對象
     * @param targetType 參數的類型
     * @param converterType 消息轉換器
     * @return true,進入,false,跳過
     */
    @Override
    public boolean supports(MethodParameter methodParameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {
        return methodParameter.hasMethodAnnotation(DecryptionAnnotation.class);
    }

    @Override
    public HttpInputMessage beforeBodyRead(HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) throws IOException {
        return inputMessage;
    }

    /**
     * 轉換之后,執行此方法,解密,賦值
     * @param body spring解析完的參數
     * @param inputMessage 輸入參數
     * @param parameter 參數對象
     * @param targetType 參數類型
     * @param converterType 消息轉換類型
     * @return 真實的參數
     */
    @SneakyThrows
    @Override
    public Object afterBodyRead(Object body, HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {

        // 獲取request
        RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
        ServletRequestAttributes servletRequestAttributes = (ServletRequestAttributes) requestAttributes;
        if (servletRequestAttributes == null) {
            throw new ParamException("request錯誤");
        }

        HttpServletRequest request = servletRequestAttributes.getRequest();

        // 獲取數據
        ServletInputStream inputStream = request.getInputStream();
        RequestData requestData = objectMapper.readValue(inputStream, RequestData.class);

        if (requestData == null || StringUtils.isBlank(requestData.getText())) {
            throw new ParamException("參數錯誤");
        }

        // 獲取加密的數據
        String text = requestData.getText();

        // 放入解密之前的數據
        request.setAttribute(CryptoConstant.INPUT_ORIGINAL_DATA, text);

        // 解密
        String decryptText = null;
        try {
            decryptText = AESUtil.decrypt(text);
        } catch (Exception e) {
            throw new ParamException("解密失敗");
        }

        if (StringUtils.isBlank(decryptText)) {
            throw new ParamException("解密失敗");
        }

        // 放入解密之后的數據
        request.setAttribute(CryptoConstant.INPUT_DECRYPT_DATA, decryptText);

        // 獲取結果
        Object result = objectMapper.readValue(decryptText, body.getClass());

        // 強制所有實體類必須繼承RequestBase類,設置時間戳
        if (result instanceof RequestBase) {
            // 獲取時間戳
            Long currentTimeMillis = ((RequestBase) result).getCurrentTimeMillis();
            // 有效期 60秒
            long effective = 60*1000;

            // 時間差
            long expire = System.currentTimeMillis() - currentTimeMillis;

            // 是否在有效期內
            if (Math.abs(expire) > effective) {
                throw new ParamException("時間戳不合法");
            }

            // 返回解密之后的數據
            return result;
        } else {
            throw new ParamException(String.format("請求參數類型:%s 未繼承:%s", result.getClass().getName(), RequestBase.class.getName()));
        }
    }

    /**
     * 如果body為空,轉為空對象
     * @param body spring解析完的參數
     * @param inputMessage 輸入參數
     * @param parameter 參數對象
     * @param targetType 參數類型
     * @param converterType 消息轉換類型
     * @return 真實的參數
     */
    @SneakyThrows
    @Override
    public Object handleEmptyBody(Object body, HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) {
        String typeName = targetType.getTypeName();
        Class<?> bodyClass = Class.forName(typeName);
        return bodyClass.newInstance();
    }
}

EncryptResponseBodyAdvice 相應自動加密;

import cn.hutool.json.JSONUtil;
import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.SneakyThrows;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;
import sun.reflect.generics.reflectiveObjects.ParameterizedTypeImpl;
import xyz.hlh.crypto.annotation.EncryptionAnnotation;
import xyz.hlh.crypto.common.entity.Result;
import xyz.hlh.crypto.common.exception.CryptoException;
import xyz.hlh.crypto.entity.RequestBase;
import xyz.hlh.crypto.util.AESUtil;

import java.lang.reflect.Type;


@ControllerAdvice
public class EncryptResponseBodyAdvice implements ResponseBodyAdvice<Result<?>> {

    @Autowired
    private ObjectMapper objectMapper;

    @Override
    public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
        ParameterizedTypeImpl genericParameterType = (ParameterizedTypeImpl)returnType.getGenericParameterType();

        // 如果直接是Result,則返回
        if (genericParameterType.getRawType() == Result.class && returnType.hasMethodAnnotation(EncryptionAnnotation.class)) {
            return true;
        }

        if (genericParameterType.getRawType() != ResponseEntity.class) {
            return false;
        }

        // 如果是ResponseEntity<Result>
        for (Type type : genericParameterType.getActualTypeArguments()) {
            if (((ParameterizedTypeImpl) type).getRawType() == Result.class && returnType.hasMethodAnnotation(EncryptionAnnotation.class)) {
                return true;
            }
        }

        return false;
    }

    @SneakyThrows
    @Override
    public Result<?> beforeBodyWrite(Result<?> body, MethodParameter returnType, MediaType selectedContentType, Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {

        // 加密
        Object data = body.getData();

        // 如果data為空,直接返回
        if (data == null) {
            return body;
        }

        // 如果是實體,并且繼承了Request,則放入時間戳
        if (data instanceof RequestBase) {
            ((RequestBase)data).setCurrentTimeMillis(System.currentTimeMillis());
        }

        String dataText = JSONUtil.toJsonStr(data);

        // 如果data為空,直接返回
        if (StringUtils.isBlank(dataText)) {
            return body;
        }

        // 如果位數小于16,報錯
        if (dataText.length() < 16) {
            throw new CryptoException("加密失敗,數據小于16位");
        }

        String encryptText = AESUtil.encryptHex(dataText);

        return Result.builder()
                .status(body.getStatus())
                .data(encryptText)
                .message(body.getMessage())
                .build();
    }
}

5.4 crypto-test

5.4.1 結構

圖片圖片

5.4.2 重要代碼

application.yml 配置文件;

spring:
  mvc:
    format:
      date-time: yyyy-MM-dd HH:mm:ss
      date: yyyy-MM-dd
  # 日期格式化
  jackson:
    date-format: yyyy-MM-dd HH:mm:ss
    time-zone: GMT+8

Teacher 實體類;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.NoArgsConstructor;
import org.hibernate.validator.constraints.Range;

import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotNull;
import java.io.Serializable;
import java.util.Date;

/**
 * @description: Teacher實體類,使用SpringBoot的validation校驗
 */
@Data
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(callSuper = true)
public class Teacher extends RequestBase implements Serializable {

    @NotBlank(message = "姓名不能為空")
    private String name;
    @NotNull(message = "年齡不能為空")
    @Range(min = 0, max = 150, message = "年齡不合法")
    private Integer age;
    @NotNull(message = "生日不能為空")
    private Date birthday;

}

TestController 測試Controller;

import org.springframework.http.ResponseEntity;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RestController;
import xyz.hlh.crypto.annotation.DecryptionAnnotation;
import xyz.hlh.crypto.annotation.EncryptionAnnotation;
import xyz.hlh.crypto.common.entity.Result;
import xyz.hlh.crypto.common.entity.ResultBuilder;
import xyz.hlh.crypto.entity.Teacher;

/**
 * @description: 測試Controller
 */
@RestController
public class TestController implements ResultBuilder {

    /**
     * 直接返回對象,不加密
     * @param teacher Teacher對象
     * @return 不加密的對象
     */
    @PostMapping("/get")
    public ResponseEntity<Result<?>> get(@Validated @RequestBody Teacher teacher) {
        return success(teacher);
    }

    /**
     * 返回加密后的數據
     * @param teacher Teacher對象
     * @return 返回加密后的數據 ResponseBody<Result>格式
     */
    @PostMapping("/encrypt")
    @EncryptionAnnotation
    public ResponseEntity<Result<?>> encrypt(@Validated @RequestBody Teacher teacher) {
        return success(teacher);
    }

    /**
     * 返回加密后的數據
     * @param teacher Teacher對象
     * @return 返回加密后的數據 Result格式
     */
    @PostMapping("/encrypt1")
    @EncryptionAnnotation
    public Result<?> encrypt1(@Validated @RequestBody Teacher teacher) {
        return success(teacher).getBody();
    }

    /**
     * 返回解密后的數據
     * @param teacher Teacher對象
     * @return 返回解密后的數據
     */
    @PostMapping("/decrypt")
    @DecryptionAnnotation
    public ResponseEntity<Result<?>> decrypt(@Validated @RequestBody Teacher teacher) {
        return success(teacher);
    }

}

責任編輯:武曉燕 來源: 一安未來
相關推薦

2023-03-06 08:49:02

加密和解密SpringBoot

2021-01-07 14:17:31

Springboot數據安全加密

2015-03-26 14:19:53

GPG加密解密

2020-09-24 10:50:53

加密解密語言hmac

2023-10-13 08:20:02

Spring線程池id

2025-03-26 08:43:17

2011-08-01 14:14:36

加密技術

2025-03-10 07:49:13

2021-05-08 05:56:15

加密OpenSSL密鑰

2018-07-30 11:56:17

解密加密開發

2012-08-29 10:13:33

StartApp分銷平臺

2021-12-28 13:54:52

加密密鑰Java

2009-12-09 17:56:27

PHP加密解密

2015-03-26 11:25:10

對稱加密加密壓縮加密解密解壓

2021-04-15 09:02:33

Python加密解密

2024-03-01 09:58:44

2021-02-01 08:00:00

vimLinux加密

2022-09-26 08:35:53

磁盤Java解密

2021-07-18 11:43:58

Linux密碼加密

2011-06-28 14:30:48

Asp.net
點贊
收藏

51CTO技術棧公眾號

国产欧美三级| 国产亚洲高清在线观看| 久久久夜色精品亚洲| 国产极品jizzhd欧美| 99自拍偷拍视频| 亚洲日本va中文字幕| 欧美色图在线视频| 丰满女人性猛交| 天天爱天天干天天操| 日韩激情中文字幕| 欧美精品福利在线| 日韩欧美黄色网址| 澳门精品久久国产| 欧美在线色视频| 可以看毛片的网址| 日本视频在线观看| 久久综合色婷婷| 91久久国产婷婷一区二区| 久久久久久久极品| 婷婷综合激情| 国产午夜精品视频免费不卡69堂| 性活交片大全免费看| 精品福利在线| 色综合色狠狠天天综合色| 欧美大片免费播放| av免费在线一区二区三区| 北条麻妃国产九九精品视频| 成人网在线视频| 久久久久久久久久成人| 一本色道精品久久一区二区三区| 久久在线免费观看视频| 91网站免费视频| 久久这里只有精品一区二区| 欧美一区二区性放荡片| 孩娇小videos精品| 欧美性片在线观看| 欧美性猛交xxxx乱大交极品| 成人免费在线网| av在线导航| 中文字幕中文字幕一区| 亚洲国内在线| 成人精品一区二区| 国产日产精品1区| 欧美中日韩免费视频| 午夜18视频在线观看| www.亚洲国产| 国产一区二区三区无遮挡 | 91久久国产精品91久久性色| 成人h动漫精品一区二区下载| 一本色道久久综合亚洲精品不卡| 久久久久久久久久国产| 麻豆91精品91久久久| 欧美高清日韩| 欧美成人免费一级人片100| 91香蕉一区二区三区在线观看 | 日本黄网站色大片免费观看| 免费大片黄在线| 国产精品久久久久影院| 正在播放91九色| av片哪里在线观看| 亚洲伊人色欲综合网| 国产天堂视频在线观看| 玖玖在线播放| 欧美性猛交xxxx黑人猛交| 无遮挡又爽又刺激的视频| 国产日韩电影| 欧美性一级生活| av免费一区二区| 欧美精品影院| 亚洲国产91精品在线观看| 久久人人爽人人人人片| 婷婷综合福利| 在线成人免费网站| 日本精品在线免费观看| 国产综合久久| 日本高清视频一区| 中文字幕人妻精品一区| 国产精品一区二区久久不卡| 国产精品久久久久久久久久久久午夜片| 亚洲国产日韩在线观看| 9人人澡人人爽人人精品| 欧美午夜精品理论片a级大开眼界| 户外极限露出调教在线视频| 亚洲欧洲在线观看av| 300部国产真实乱| 伊人久久视频| 欧美精品v国产精品v日韩精品| 亚洲一区日韩精品| 91精品国产自产精品男人的天堂| 日韩风俗一区 二区| 久久久久久久毛片| 欧美.www| 国产成人综合精品在线| 96日本xxxxxⅹxxx17| 成人午夜电影小说| 日韩久久在线| 国内在线视频| 欧美日韩在线精品一区二区三区激情 | 黄视频在线观看网站| 亚洲激情男女视频| 国产日韩成人内射视频| 久久久久久久久久久久电影| 亚洲欧美日韩国产中文| 丝袜美腿小色网| 蜜桃视频一区| yy111111少妇影院日韩夜片| 经典三级在线| 亚洲国产日日夜夜| 艹b视频在线观看| 欧美挤奶吃奶水xxxxx| 精品国产自在精品国产浪潮| 精品国产免费观看| 国产呦精品一区二区三区网站| 精品综合在线| fc2ppv国产精品久久| 91国模大尺度私拍在线视频| 亚洲图片综合网| 欧美.www| 91系列在线观看| 国产高清一级毛片在线不卡| 精品美女国产在线| 成年人性生活视频| 999精品色在线播放| 日本午夜在线亚洲.国产| 亚洲欧美另类视频| 亚洲视频一区二区在线观看| 手机看片福利盒子久久| 美日韩黄色大片| 欧美大尺度在线观看| 在线观看色网站| 国产偷国产偷亚洲高清人白洁| 国产v片免费观看| 99ri日韩精品视频| 久久6精品影院| 国产偷拍一区二区| **网站欧美大片在线观看| 亚洲欧美久久久久| 中文精品一区二区| 欧美自拍视频在线观看| 手机看片国产1024| 亚洲最新视频在线播放| 国产黑丝在线视频| 欧美在线影院| caoporn国产精品免费公开| av片在线观看| 精品三级在线看| 久久久久久激情| 成人精品电影在线观看| 大陆av在线播放| 日本精品影院| 日本欧美精品在线| 黑人与亚洲人色ⅹvideos| 色噜噜狠狠色综合欧洲selulu| 色噜噜日韩精品欧美一区二区| 久久久精品日韩| 日韩区国产区| 日韩美香港a一级毛片| 久久精品91久久香蕉加勒比| 国产女人高潮的av毛片| 亚洲激情自拍视频| 精品人妻一区二区三区日产| 一本不卡影院| 午夜欧美一区二区三区免费观看| 成人深夜福利| 日韩视频免费在线观看| av手机免费看| 婷婷久久综合九色国产成人| 一区二区精品免费| 蜜桃精品在线观看| 黄色一级片av| 欧美成人一区在线观看| 国产成人精品视| 日本在线观看www| 日韩区在线观看| www..com国产| 欧美国产精品一区| 国产探花在线观看视频| 在线成人国产| 日韩一区二区三区高清| **日韩最新| 91黑丝在线观看| 日本在线视频观看| 亚洲国产精品电影| 中文字幕在线观看精品| 一区二区三区丝袜| 丰满少妇一区二区三区| 久久精品国产成人一区二区三区 | 羞羞答答国产精品www一本| 日韩精品欧美专区| 日韩一区免费| 国产精品久久久久久久久久99| 成人a在线视频免费观看| 亚洲国产精品成人精品| 国产又黄又爽视频| 精品久久中文字幕久久av| 欧日韩不卡视频| av成人动漫在线观看| 九九热免费在线观看| 亚洲少妇在线| 91国在线高清视频| 欧美在线色图| 国产一区二区三区奇米久涩| 欧美91在线|欧美| 97人洗澡人人免费公开视频碰碰碰| yiren22亚洲综合伊人22| 精品国产一区二区三区久久久蜜月| 国产精品sm调教免费专区| 亚洲18色成人| 最新一区二区三区| 国产欧美日韩在线看| 国产xxxxxxxxx| 国产剧情一区在线| 手机视频在线观看| 久久久久看片| 波多野结衣综合网| 国内精品福利| 久久观看最新视频| 我不卡伦不卡影院| 亚洲一区二区三区加勒比| 婷婷亚洲精品| 国产专区一区二区| 在线视频亚洲欧美中文| 成人信息集中地欧美| 成人免费黄色| 国产精品9999| 欧美成人精品一区二区男人小说| 91国偷自产一区二区三区的观看方式 | 91一区一区三区| 女人扒开腿免费视频app| 美美哒免费高清在线观看视频一区二区| 男女私大尺度视频| 欧美理论在线| 一本色道久久88亚洲精品综合| 久久久影院免费| 亚洲毛片aa| 精品一区二区三区的国产在线观看| 精选一区二区三区四区五区| 美女视频免费精品| 精品免费日产一区一区三区免费| jizz性欧美23| 国产精品视频免费一区| 黑色丝袜福利片av久久| 国产伦精品一区二区三区四区免费| 成人在线tv视频| 国产在线精品日韩| 牛牛影视久久网| 欧美日韩亚洲一区二区三区在线观看| 综合伊思人在钱三区| 欧美视频小说| 日韩精品一区二区三区免费观影 | 亚洲男人影院| 人妻内射一区二区在线视频| 亚洲综合国产| www.99av.com| 久久精品国产99久久6 | 日韩最新av| 国产精品视频福利| 亚洲精华一区二区三区| 日韩.欧美.亚洲| 99精品视频在线观看播放| 亚洲制服欧美久久| 欧美fxxxxxx另类| 国产美女主播在线播放| 国产精品入口| 五月天婷婷激情视频| 久久er99热精品一区二区| 亚欧精品在线视频| 成人av资源站| 一区二区伦理片| 亚洲欧洲精品一区二区精品久久久 | 成人在线免费视频观看| 椎名由奈jux491在线播放 | 天天色综合天天色| 国产在线麻豆精品观看| av不卡中文字幕| 久久一二三国产| 亚洲波多野结衣| 亚洲成在线观看| 亚洲国产无线乱码在线观看| 日韩精品一区二区三区视频播放| 手机在线精品视频| 亚洲日韩中文字幕| 91三级在线| 国产福利成人在线| 日韩高清在线观看一区二区| 蜜桃免费一区二区三区| 国产精品久久久久蜜臀| 久在线观看视频| 久久99久久久久久久久久久| 亚洲视频在线播放免费| 国产精品免费视频一区| 久久精品久久精品久久| 欧美午夜精品久久久| 韩国视频理论视频久久| 精品在线视频观看| 欧美日韩在线看| ,一级淫片a看免费| 亚洲精品电影网站| 免费看a在线观看| 91av视频在线| 警花av一区二区三区| 欧美日韩日本网| 亚洲天堂男人| 欧美精品久久久久久久久25p| 成人永久免费视频| 北条麻妃在线观看视频| 日韩欧美aaa| 高清一区二区三区四区| 日韩在线观看视频免费| 波多野结衣久久精品| 成人动漫在线视频| 国产精品毛片久久| 日韩中文字幕免费在线| 成人av在线一区二区三区| 侵犯稚嫩小箩莉h文系列小说| 色综合天天综合狠狠| 人妻精品无码一区二区| 美女撒尿一区二区三区| 国产成人免费精品| 欧美日韩一区二| 欧美一级视频| 朝桐光av一区二区三区| 一区二区三区在线视频观看| 96亚洲精品久久久蜜桃| 伊人青青综合网站| 午夜精品久久久久久久久久蜜桃| 国产偷久久久精品专区| 亚洲欧美伊人| 91人妻一区二区三区| 亚洲欧美综合网| 亚洲图片中文字幕| 日韩在线欧美在线| 成人综合网站| 日韩一区二区电影在线观看| 日韩电影在线免费观看| 欧洲女同同性吃奶| 色偷偷成人一区二区三区91| 青青青草原在线| 欧美在线www| 亚洲香蕉视频| 青青在线免费观看视频| 国产校园另类小说区| 黄色网址中文字幕| 国产一区二区三区毛片| 成人四虎影院| 一本久道久久综合| 激情图区综合网| 欧美激情精品久久久久久免费| 欧美精品1区2区| 二区在线播放| 成人蜜桃视频| 一本色道久久综合一区| 熟女少妇一区二区三区| 91官网在线免费观看| 97最新国自产拍视频在线完整在线看| 国产精品久久中文| 天天天综合网| 熟妇女人妻丰满少妇中文字幕| 一区二区三区在线高清| 欧美在线 | 亚洲| 日韩美女在线观看| 久久一区91| 下面一进一出好爽视频| 亚洲成人免费视频| 蜜桃视频在线观看网站| 91精品久久久久久久久久久| 影视亚洲一区二区三区| 香港三日本8a三级少妇三级99| 日韩欧美国产免费播放| 97最新国自产拍视频在线完整在线看| 91免费在线视频| 亚洲二区视频| 中文字幕在线观看免费高清| 欧美一区二区三区色| 丁香花高清在线观看完整版| 欧美日韩精品免费看| 精品一区二区三区不卡| 久久久久久久福利| 亚洲欧美日韩另类| crdy在线观看欧美| 日本韩国欧美在线观看| 亚洲国产精品99久久久久久久久 | 亚洲午夜精品久久久久久浪潮| 国产乱码精品一区二区三区五月婷| 国产在线观看免费av| 亚洲人成人99网站| 国产日韩中文在线中文字幕| 欧美性大战久久久久xxx| 国产精品成人免费精品自在线观看 | 日本在线观看一区| 国模娜娜一区二区三区| 日韩女同强女同hd| 日韩中文字幕不卡视频| 粉嫩的18在线观看极品精品| 久久久久久久久久久免费视频| 亚洲精品视频免费看| 免费在线超碰| 国产精品99久久久久久久 | 成人午夜黄色影院| 羞羞视频在线观看欧美|