精品欧美一区二区三区在线观看 _久久久久国色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技術棧公眾號

欧美不卡三区| 久久久久中文字幕2018| www.五月天色| 后进极品白嫩翘臀在线播放| 91首页免费视频| 国产精品色视频| 久久婷婷国产麻豆91| 亚洲人成网77777色在线播放| 欧美探花视频资源| 亚洲爆乳无码精品aaa片蜜桃| 日本高清中文字幕二区在线| 精品亚洲欧美一区| 欧美专区在线观看| 911国产在线| 午夜先锋成人动漫在线| 欧美精品三级日韩久久| 国产深夜男女无套内射| 久久久久久国产精品免费无遮挡| av在线播放一区二区三区| 国产精品青草久久久久福利99| 免费无遮挡无码永久在线观看视频 | 2019中文字幕在线| 黄色一级大片在线免费观看| 伊人久久大香线蕉综合网站| 日韩欧美亚洲另类制服综合在线| www.涩涩涩| 欧美少妇精品| 亚洲综合色成人| 自拍视频一区二区三区| 毛片网站在线| 99久久99久久精品国产片果冻| 国产精品自拍视频| 中文字幕免费高清网站| 亚洲美女啪啪| 欧美国产亚洲精品久久久8v| 超碰人人人人人人人| 性人久久久久| 亚洲精品v欧美精品v日韩精品| av在线免费观看不卡| 成人不卡视频| 欧美午夜www高清视频| 国产一级做a爰片久久毛片男| 99re在线视频| 久久久久久久久久久久久女国产乱| 国产精品12| 精品国产亚洲av麻豆| 精品在线观看免费| 国产精品自拍小视频| 波多野结衣小视频| 国产亚洲综合精品| 97热精品视频官网| 日韩成人免费在线视频| 欧美日韩一区二区国产| 另类图片亚洲另类| 国产一二三区精品| 一本一道久久综合狠狠老| xxx成人少妇69| 一级性生活免费视频| 欧美aaaaaaaaaaaa| 精品国偷自产在线| 五月天激情丁香| 欧美在线首页| 色综合久久精品亚洲国产| 欧美精品一级片| 亚洲第一区色| 91精品国产91久久久久久吃药| 黄色一级片免费看| 国产精品久久久免费| 欧洲日韩成人av| 国产成人精品一区二区色戒| 久久成人av少妇免费| 成人在线一区二区| 成人午夜福利视频| 成av人片一区二区| 欧美精品人人做人人爱视频| 国际av在线| 亚洲欧洲av一区二区三区久久| 国产日韩第一页| 日本不卡影院| 色综合久久中文字幕| 国产精品无码av无码| 欧美a视频在线| 日韩欧美另类在线| 在线免费观看a级片| 精品国产成人| 成人97在线观看视频| 日本熟妇乱子伦xxxx| 玖玖国产精品视频| 91国产丝袜在线放| 天堂av中文在线资源库| 中文天堂在线一区| 国产真人做爰毛片视频直播| 亚洲最大网站| 欧美一级高清大全免费观看| 艳妇乳肉豪妇荡乳xxx| 狠狠做深爱婷婷综合一区| 久热精品视频在线观看| 六月丁香婷婷综合| 狠狠色伊人亚洲综合成人| 国产精品一区免费观看| av一本在线| 亚洲丰满少妇videoshd| 爱情岛论坛亚洲首页入口章节| 天堂精品久久久久| 亚洲图片在区色| 久久成人在线观看| 毛片基地黄久久久久久天堂| 高清一区二区三区视频| 成人网视频在线观看| 夜夜爽夜夜爽精品视频| 久久久精品三级| 凹凸av导航大全精品| 一区二区中文字幕| 可以免费在线观看的av| 国内精品久久久久影院一蜜桃| 久久精品国产综合精品| 亚洲羞羞网站| 欧美日韩国产精品自在自线| japanese在线观看| 一个色综合网| 国产日韩欧美在线| 深夜福利视频在线免费观看| 一区二区三区小说| 999在线观看| 国产91久久精品一区二区| 欧美国产日本在线| 国产精品色综合| 国产精品免费视频观看| av观看免费在线| 黄色欧美在线| 欧美黑人xxx| av资源免费看| 自拍偷拍亚洲激情| 嫩草av久久伊人妇女超级a| 欧美一区二区三区久久| 欧美激情三级免费| www日本高清| 亚洲三级电影网站| 亚洲精品免费一区亚洲精品免费精品一区 | 欧美日韩免费观看一区三区| 免费黄色在线视频| 亚洲一区二区三区免费在线观看| 国产精品美女xx| 免费电影网站在线视频观看福利| 884aa四虎影成人精品一区| 污污视频网站在线免费观看| 日韩av在线免费观看不卡| 久久综合九九| 日韩电影免费观| 亚洲人成自拍网站| 日韩黄色一级视频| 日本一区二区三区国色天香| 久章草在线视频| 国产一区二区三区不卡视频网站| 日韩免费观看av| 国产黄色在线| 欧美疯狂性受xxxxx喷水图片| 激情无码人妻又粗又大| 久久66热re国产| 小泽玛利亚av在线| 成人爽a毛片| 18久久久久久| www免费网站在线观看| 欧美精品在欧美一区二区少妇| 呻吟揉丰满对白91乃国产区| 精品系列免费在线观看| 大地资源网在线观看免费官网| 136国产福利精品导航网址应用| 欧美激情国内偷拍| 秋霞网一区二区| 欧美日韩精品在线| 日韩视频在线观看免费视频| 久久99久久精品| 国产 国语对白 露脸| 欧美激情15p| 国产精品亚洲片夜色在线| 国产精品一区二区三区视频网站| 欧美xxxxxxxx| 特级做a爱片免费69| 国产精品午夜春色av| 久久精品一卡二卡| 国产精品一区毛片| 日韩中文一区二区三区| 国产成人免费av一区二区午夜| 欧美大片免费观看| 黄色片在线播放| 欧美一区二区三区在线观看 | 欧美13一16娇小xxxx| 日韩精品自拍偷拍| 国产精品免费精品一区| 综合色天天鬼久久鬼色| 亚洲观看黄色网| 美女性感视频久久| 国产无限制自拍| 久久久综合色| 精品久久久久久中文字幕动漫| 国内精品伊人| 亚州国产精品久久久| 免费黄色在线看| 日韩精品极品在线观看| 国产原创中文av| 色综合天天天天做夜夜夜夜做| 97精品在线播放| 久久综合色8888| 日批视频免费看| 美女脱光内衣内裤视频久久网站| 久久久久久久久久久99| 999视频精品| 日本公妇乱淫免费视频一区三区| 久久久久毛片免费观看| 国产精品成人aaaaa网站| 日本在线视频网址| 日韩最新av在线| 日本不卡视频一区二区| 日韩欧美综合在线| 亚洲图片欧美在线| 日韩欧美精品网址| 久久网中文字幕| 亚洲欧洲av一区二区三区久久| 男生草女生视频| av午夜一区麻豆| 无码国产精品一区二区高潮| 免费在线观看精品| 91传媒久久久| 99热免费精品在线观看| 黄色一级片国产| 亚洲色图欧美| 自拍偷拍一区二区三区| 国产99精品| 欧美一区二区在线| 羞羞色国产精品网站| 精品视频在线观看| 国产精品视屏| 91精品国产一区二区三区动漫 | 欧美激情20| 欧美激情视频在线免费观看 欧美视频免费一| 成人18在线| 国产一区二区动漫| 国产中文字幕在线视频| 国产丝袜精品视频| 欧美日韩在线精品一区二区三区激情综| 精品国产1区二区| 成人毛片视频免费看| 欧美精品一区二区三区很污很色的| 国产a级免费视频| 欧美一区二区啪啪| 国产日韩在线观看一区| 8v天堂国产在线一区二区| 在线观看视频二区| 欧美日韩不卡在线| 一级片在线观看视频| 这里只有精品电影| 国产成人久久精品77777综合| 日韩一区二区三区免费看| 国产伦精品一区二区三区免.费| 69堂精品视频| 国产福利免费视频| 亚洲а∨天堂久久精品喷水| 五月天婷婷视频| 亚洲另类图片色| av在线女优影院| 精品国产欧美一区二区五十路 | 日本美女一级片| 日韩电影在线观看中文字幕| 美丽的姑娘在线观看免费动漫| 亚洲色图狂野欧美| 麻豆av在线导航| 九色精品美女在线| 日韩在线伦理| 国产精品视频精品视频| 国产高清日韩| 国产区日韩欧美| 精品福利久久久| 综合色婷婷一区二区亚洲欧美国产| 91精品国产麻豆国产在线观看| 成人午夜免费剧场| 亚洲影音先锋| 日韩一级免费片| 成人网男人的天堂| 色噜噜日韩精品欧美一区二区| 国产精品网站在线| 麻豆亚洲av熟女国产一区二| 欧美性色19p| 一区二区三区午夜| 亚洲国产精品一区二区久| 高清国产福利在线观看| 美女性感视频久久久 | 国产精品久久97| 涩爱av色老久久精品偷偷鲁| 久久精品人成| 午夜精品毛片| 91av在线免费播放| 粉嫩av亚洲一区二区图片| 波多野结衣av在线免费观看| 亚洲欧洲成人精品av97| 欧美videossex极品| 欧美一区二区视频观看视频 | 久久成人亚洲精品| 在线观看爽视频| 亚洲精品日韩av| 国语产色综合| 激情五月宗合网| 国产主播一区二区三区| 中日韩精品一区二区三区 | 强开小嫩苞一区二区三区网站 | 亚洲一区二区三区无吗| 精品人妻一区二区三区四区在线 | 精品欧美日韩在线| 在线精品国产| 9久久婷婷国产综合精品性色 | 日本精品在线中文字幕| 懂色av一区二区三区在线播放| 成人影院天天5g天天爽无毒影院| 无码日本精品xxxxxxxxx| 男女性色大片免费观看一区二区| 最近中文字幕无免费| 一区二区三区日韩精品视频| 一级特黄免费视频| 亚洲黄色av网站| 污污视频在线看| 成人av番号网| 精品国产日韩欧美| 日韩av资源在线| 北条麻妃国产九九精品视频| 少妇久久久久久被弄高潮| 欧美日韩亚洲综合在线| 久久综合九色综合久| 91高潮精品免费porn| 福利在线一区| 91成人综合网| 成人综合婷婷国产精品久久免费| www.5588.com毛片| 欧美日本高清视频在线观看| 国产精品视频一区二区久久| 欧美中文在线字幕| 全球av集中精品导航福利| 91成人综合网| 成人在线视频首页| 久草视频精品在线| 精品捆绑美女sm三区| 影院在线观看全集免费观看| 91社区国产高清| 亚洲综合激情在线| 无人码人妻一区二区三区免费| 亚洲美女一区二区三区| 国产女人高潮时对白| 久久精品国产一区二区电影| 日日夜夜一区| 一区二区三区三区在线| 精品一区免费av| 免费在线观看a级片| 91精品国产高清一区二区三区| 国产精品实拍| 国产高清精品一区二区| 国产综合网站| 特级西西人体wwwww| 色综合久久综合中文综合网| 可以在线观看的av| 国产精品久久久久久中文字| 日韩欧美高清在线播放| 在线观看日本www| 亚洲韩国精品一区| 亚洲日本香蕉视频| 日本乱人伦a精品| 日韩综合在线| 日本少妇xxxx软件| 五月激情综合婷婷| 国产高清视频在线播放| 成人久久久久久| 狠狠噜噜久久| 日韩av在线看免费观看| 欧美在线不卡一区| 岛国中文字幕在线| 国产精品视频福利| 肉肉av福利一精品导航| 亚洲波多野结衣| 亚洲精品一线二线三线无人区| 免费成人在线电影| 日韩一区二区三区高清| 国产麻豆一精品一av一免费| 日韩经典在线观看| 伊人av综合网| 一区二区在线视频观看| 欧美在线观看视频网站| 自拍av一区二区三区| 天堂中文资源在线观看| 国产精品视频久久久久| 欧美体内she精视频在线观看| 亚洲做受高潮无遮挡| 欧美福利电影网| 在线天堂资源www在线污| 亚洲不卡1区| 狠狠色狠狠色综合| aaaaaa毛片| 欧美国产视频一区二区| 欧美亚洲国产激情| 少妇搡bbbb搡bbb搡打电话| 欧美影院精品一区| brazzers在线观看| 影音先锋亚洲视频|