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

Spring Boot 接口數據加解密,太方便了!

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

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

圖片圖片

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

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

需求解析:

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

接口加密解密

2022-12-14 09:06:58

接口Spring解密

2025-07-03 02:11:00

Swing開發C++

2025-07-21 05:00:00

if-elseV1版本

2022-09-14 10:16:12

MyBatis加密解密

2025-05-14 04:00:00

2022-07-15 14:26:36

開源工具IP

2024-08-13 10:36:25

SpringScrew數據庫

2021-09-24 15:00:26

微信PC電腦移動應用

2025-07-08 02:12:00

2021-09-24 09:59:59

復制粘貼PythonPDF

2025-08-26 04:00:00

2025-03-03 00:00:55

Spring文件下載開發

2019-07-24 10:50:56

Python 開發編程語言

2024-04-15 00:00:02

OpenAI模型性能

2021-03-09 13:18:53

加密解密參數

2022-06-04 12:25:10

解密加密過濾器

2022-09-14 18:23:01

工具加解密接口

2011-08-01 14:36:06

加密RSA

2021-02-08 11:46:17

Python自動化郵件
點贊
收藏

51CTO技術棧公眾號

波多野结衣片子| 欧美视频在线观看网站| 在线免费日韩av| 国产精品tv| 日本道精品一区二区三区| 伊人久久婷婷色综合98网| 国产熟女一区二区三区五月婷| 伊人精品成人久久综合软件| 亚洲美女av电影| 国产黑丝在线视频| 成人勉费视频| 亚洲精品国产精华液| 欧美裸体网站| а√中文在线资源库| 久久黄色影院| 久久久久久av| 青青青视频在线免费观看| 9999久久久久| 在线不卡免费av| 人妻熟女一二三区夜夜爱| 永久免费网站在线| 国产日韩欧美亚洲| 精品久久精品久久| www.黄色小说.com| 免费人成在线不卡| 欧美最顶级的aⅴ艳星| 国产少妇在线观看| 日韩欧美字幕| 亚洲一级免费视频| 黄色录像a级片| 无码国模国产在线观看| 欧美日韩一区二区三区高清 | 日本高清免费观看| 91精品影视| 黄色精品一区二区| 国产1区2区3区中文字幕| 欧美性videos| 中文字幕免费不卡| 麻豆av福利av久久av| 亚洲成人777777| 久久精品国产免费| 国产精品久久国产精品99gif| 久久精品国产亚洲AV无码麻豆| 国产精品国产三级国产在线观看 | 国产乱码精品一区二区三区不卡| 一卡二卡在线观看| 美女www一区二区| 国产精品成熟老女人| 中文字幕av影院| 国产欧美欧美| 欧美性一区二区三区| 亚洲国产精一区二区三区性色| 中文一区一区三区免费在线观看| 中文字幕无线精品亚洲乱码一区 | 免费观看成年人视频| 国产激情视频一区二区三区欧美| 91精品视频观看| 国产乱淫a∨片免费视频| 精品一区精品二区高清| 成人妇女淫片aaaa视频| 国产精品国产av| 精品在线一区二区| 成人h视频在线观看播放| 91成人一区二区三区| 精品制服美女丁香| 91免费视频网站| 91视频在线视频| 日韩avvvv在线播放| 国产精品成人aaaaa网站| 波多野结衣视频网址| 蜜桃视频一区二区三区| 国产在线拍偷自揄拍精品| 一卡二卡三卡在线| 成人妖精视频yjsp地址| 国内一区二区在线视频观看| 视频一区二区三区在线看免费看| 久久久久久久久99精品| 少妇免费毛片久久久久久久久| 成人77777| 亚洲老妇xxxxxx| 日韩欧美不卡在线| 影视一区二区三区| 777精品伊人久久久久大香线蕉| 男插女视频网站| 欧美顶级毛片在线播放| 伊人久久男人天堂| 最新一区二区三区| 在线日韩电影| 国产精品香蕉av| 亚洲成人av综合| 国产视频一区二区在线| 懂色av一区二区三区四区五区| 欧美色图天堂| 色哟哟一区二区| 奇米777在线| 久久久免费毛片| 在线午夜精品自拍| 久久黄色小视频| 三级精品在线观看| 91免费在线观看网站| 色视频在线观看免费| 最新国产の精品合集bt伙计| 少妇人妻大乳在线视频| 成人黄页网站视频| 亚洲黄色www网站| 永久免费未视频| 性色一区二区三区| 成人性色av| 色网站在线看| 日韩欧美在线视频| 波多野结衣免费观看| 国产99精品一区| 久久久久久91| 国产免费av电影| 国产日韩欧美不卡在线| 日韩精品在线中文字幕| 日韩三级一区| 亚洲视频777| 日韩av片在线播放| 激情另类小说区图片区视频区| 蜜桃成人在线| 激情av在线| 欧美高清视频一二三区 | 国产精九九网站漫画| 久久免费精品视频在这里| 国产91av在线| 欧美 日韩 综合| 亚洲激情图片qvod| www.日本一区| 精品国产精品久久一区免费式 | 欧美日韩免费观看视频| 亚洲第一中文字幕| 九九视频在线观看| 国产中文一区二区三区| 一区不卡字幕| 欧美成人aaa| 国产午夜精品视频| 亚洲免费在线视频观看| 成人av在线一区二区三区| 蜜桃视频一区二区在线观看| 福利一区三区| 久久视频在线视频| 国产免费叼嘿网站免费| 亚洲欧美在线视频| 成年人三级黄色片| 97久久视频| 成人淫片在线看| 麻豆系列在线观看| 91麻豆精品国产91久久久更新时间 | 97久久精品人人做人人爽50路| 国产xxxx振车| 高清欧美性猛交xxxx黑人猛| 欧美激情国产日韩精品一区18| 99久久婷婷国产一区二区三区| 亚洲视频一区二区在线| 97人人模人人爽人人澡| 欧美精品aa| 国产一级特黄a大片99| 电影k8一区二区三区久久| 欧美精品一区二区久久婷婷| 日韩乱码在线观看| 久久综合九色综合97婷婷女人| 国产无套内射久久久国产| 蜜臀91精品国产高清在线观看| 国产成人av网| 日本www在线| 日韩欧美一区在线观看| 日本午夜小视频| 91麻豆精品秘密| 欧美精品aaaa| 亚洲成av人片乱码色午夜| 91成人免费在线观看| 成人免费图片免费观看| 亚洲免费成人av电影| 怡春院在线视频| 一区二区视频在线看| 中文文字幕文字幕高清| 日韩av中文字幕一区二区三区| 一区二区三区在线视频看| 欧美h版在线观看| 97人洗澡人人免费公开视频碰碰碰| 青青色在线视频| 欧美日韩国产综合一区二区三区 | 国产综合欧美| 欧美中日韩免费视频| 激情久久99| 久久久欧美一区二区| 国产女人在线视频| 91精品国产91久久综合桃花| 日韩视频免费观看高清| 国产欧美综合在线观看第十页| 国产精品久久久久久9999| 国产欧美午夜| 自拍另类欧美| 亚洲图片久久| 91免费看蜜桃| 91欧美精品| 久久人91精品久久久久久不卡| 电影av在线| 亚洲成人黄色网址| 一区二区精品视频在线观看| 亚洲成a人在线观看| 在线观看天堂av| 91在线一区二区| 天天干天天曰天天操| 日本欧美在线看| 国产精品国产亚洲精品看不卡| 国产精品传媒精东影业在线 | 久久久久久久久国产一区| 精品乱码一区二区三区| 国产电影一区二区| 国产成人精品在线| av中文字幕电影在线看| www.xxxx欧美| 黄网在线观看| 日韩精品日韩在线观看| 精品国产18久久久久久| 欧洲视频一区二区| 国产欧美日韩另类| 一区二区在线看| 欧美一级片在线视频| 国产三级久久久| 人妻丰满熟妇aⅴ无码| 国产成人免费高清| 欧美一级小视频| 日本 国产 欧美色综合| wwwxxx黄色片| 午夜在线观看免费一区| 国产精品一线二线三线| 午夜视频精品| 蜜桃网站在线观看| 91高清一区| 日本高清xxxx| 亚洲免费二区| 99精品一级欧美片免费播放| 日韩国产一区| 五月婷婷综合色| 欧美精品一区二区三区精品| 欧美日韩三区四区| 婷婷成人综合| 欧美日韩精品久久| 最新国产一区| 视频一区亚洲| 大胆日韩av| 国产999精品视频| 日本中文在线观看| 亚洲色在线视频| 国产福利在线看| 亚洲日韩中文字幕| 黄色美女网站在线观看| 亚洲人成电影网站色| 男男激情在线| 亚洲新声在线观看| www.av在线| 日韩三级成人av网| 欧洲日本在线| 欧美成人中文字幕| 成人三级小说| 91wwwcom在线观看| 亚洲精品国产精品国产| 日本一区二区在线播放| 桃花岛成人影院| 国产精品美女免费看| 亚洲一区有码| 成人看片在线| 亚洲高清极品| 亚洲欧美日韩精品在线| 国产大片一区| 国产青草视频在线观看| 国产欧美欧美| 伊人国产在线视频| 国产乱码精品一区二区三区五月婷| 中国男女全黄大片| 99v久久综合狠狠综合久久| 在线不卡av电影| 中文字幕一区在线| 久久久久久久极品内射| 精品女厕一区二区三区| 夜夜爽妓女8888视频免费观看| 欧美三级日韩在线| 99热这里精品| 亚洲剧情一区二区| 麻豆视频在线| 午夜精品久久久久久久99黑人| 欧美18—19sex性hd| 91精品国产综合久久香蕉最新版| 中文字幕一区日韩精品| 欧美黄色直播| 欧美伊人久久| 免费男同深夜夜行网站| 国产一区二区精品久久99| 亚洲图片欧美另类| 中文字幕免费不卡| 日本特黄一级片| 欧美日韩一区二区三区高清 | 一区二区乱码| 91久久精品国产91久久性色| 韩国女主播一区二区三区| 天天综合色天天综合色hd| 国产精品chinese| 成年人在线观看视频免费| 国产成人午夜视频| 国产在线综合视频| 亚洲午夜激情网站| 一本色道久久综合亚洲| 日韩av中文字幕在线播放| 黄色网页在线免费观看| 欧美与黑人午夜性猛交久久久| 自拍偷拍亚洲| 日本亚洲欧洲精品| 99国产精品99久久久久久粉嫩| 最新天堂在线视频| 91蜜桃网址入口| 国产一级特黄a高潮片| 在线观看91av| 国产日韩精品在线看| 午夜精品久久久久久久久久久久久| 婷婷丁香久久| 日韩伦理一区二区三区av在线| 激情成人亚洲| 中文国产在线观看| 国产精品女主播在线观看| 九一国产在线观看| 欧美大片在线观看一区| 日本精品在线| 国产精品视频一区二区高潮| 日韩av三区| 韩日视频在线观看| 国产剧情一区二区| 老熟妇高潮一区二区三区| 欧美婷婷六月丁香综合色| 免费国产在线视频| 7777kkkk成人观看| 精品国产午夜肉伦伦影院| 强开小嫩苞一区二区三区网站| 另类小说综合欧美亚洲| 卡一卡二卡三在线观看| 色噜噜久久综合| 嫩草在线播放| 日韩女优在线播放| 奇米亚洲欧美| 日本xxxxxxx免费视频| 久久这里只有精品视频网| 欧美日韩精品区| 精品一区二区三区电影| 欲香欲色天天天综合和网| 久久99欧美| 久久天堂成人| 成年人在线免费看片| 欧洲在线/亚洲| 天天在线视频色| 成人久久一区二区| 中文字幕亚洲综合久久五月天色无吗''| www.色欧美| 一区二区视频免费在线观看| 国产免费叼嘿网站免费| 久久成人av网站| 99精品在免费线中文字幕网站一区 | 精品三级久久久久久久电影聊斋| 奇门遁甲1982国语版免费观看高清| 网友自拍一区| 欧美精品第三页| 亚洲国产高清在线观看视频| 亚洲一级在线播放| 久久成人免费视频| 97精品久久| 欧美 日本 亚洲| 欧美国产成人在线| 国产一区二区在线播放视频| 九九久久久久99精品| 国产图片一区| 嫩草av久久伊人妇女超级a| 国产精品天美传媒| 999久久久久| 97香蕉超级碰碰久久免费软件| 伊人春色精品| 日本国产一级片| 亚洲国产成人porn| 可以在线观看的av网站| 成人免费视频网址| 亚洲欧洲一区二区天堂久久| 久久久久亚洲av成人无码电影 | 人人妻人人澡人人爽久久av| 热久久免费国产视频| 日韩欧美视频| 四虎成人免费视频| 色综合久久久网| 黄网站在线播放| 精品中文字幕一区| 久久99国产精品免费网站| 免费无码毛片一区二区app| 亚洲精品视频在线观看视频| 中文字幕日本一区| 丰满爆乳一区二区三区| 国产精品免费av| 五月婷中文字幕| 91pron在线| 美女性感视频久久| 日本熟妇成熟毛茸茸| 少妇久久久久久|