点击上方 好好学java ,选择 星标 公众号

重磅资讯、干货,第一时间送达

今日推荐:腾讯推出高性能 RPC 开发框架

个人原创100W+访问量博客:点击前往,查看更多

在日常开发中,总会接触到各种接口。前后端数据传输接口,第三方业务平台接口。一个平台的前后端数据传输接口一般都会在内网环境下通信,而且会使用安全框架,所以安全性可以得到很好的保护。这篇文章重点讨论一下提供给第三方平台的业务接口应当如何设计?我们应该考虑哪些问题?

主要从以上三个方面来设计一个安全的API接口。

一 安全性问题

安全性问题是一个接口必须要保证的规范。如果接口保证不了安全性,那么你的接口相当于直接暴露在公网环境中任人蹂躏。

1.1 调用接口的先决条件-token

获取token一般会涉及到几个参数appidappkeytimestampnoncesign。我们通过以上几个参数来获取调用系统的凭证。

appidappkey可以直接通过平台线上申请,也可以线下直接颁发。appid是全局唯一的,每个appid将对应一个客户,appkey需要高度保密。

timestamp是时间戳,使用系统当前的unix时间戳。时间戳的目的就是为了减轻DOS攻击。防止请求被拦截后一直尝试请求接口。服务器端设置时间戳阀值,如果请求时间戳和服务器时间超过阀值,则响应失败。

nonce是随机值。随机值主要是为了增加sign的多变性,也可以保护接口的幂等性,相邻的两次请求nonce不允许重复,如果重复则认为是重复提交,响应失败。

sign是参数签名,将appkeytimestampnonce拼接起来进行md5加密(当然使用其他方式进行不可逆加密也没问题)。

token,使用参数appidtimestampnoncesign来获取token,作为系统调用的唯一凭证。token可以设置一次有效(这样安全性更高),也可以设置时效性,这里推荐设置时效性。如果一次有效的话这个接口的请求频率可能会很高。token推荐加到请求头上,这样可以跟业务参数完全区分开来。

1.2 使用POST作为接口请求方式

一般调用接口最常用的两种方式就是GET和POST。两者的区别也很明显,GET请求会将参数暴露在浏览器URL中,而且对长度也有限制。为了更高的安全性,所有接口都采用POST方式请求。

1.3 客户端IP白名单

ip白名单是指将接口的访问权限对部分ip进行开放。这样就能避免其他ip进行访问攻击,设置ip白名单比较麻烦的一点就是当你的客户端进行迁移后,就需要重新联系服务提供者添加新的ip白名单。设置ip白名单的方式很多,除了传统的防火墙之外,spring cloud alibaba提供的组件sentinel也支持白名单设置。为了降低api的复杂度,推荐使用防火墙规则进行白名单设置。

1.4 单个接口针对ip限流

限流是为了更好的维护系统稳定性。使用redis进行接口调用次数统计,ip+接口地址作为key,访问次数作为value,每次请求value+1,设置过期时长来限制接口的调用频率。

1.5 记录接口请求日志

使用aop全局记录请求日志,快速定位异常请求位置,排查问题原因。

1.6 敏感数据脱敏

在接口调用过程中,可能会涉及到订单号等敏感数据,这类数据通常需要脱敏处理,最常用的方式就是加密。加密方式使用安全性比较高的RSA非对称加密。非对称加密算法有两个密钥,这两个密钥完全不同但又完全匹配。只有使用匹配的一对公钥和私钥,才能完成对明文的加密和解密过程。

二 幂等性问题

幂等性是指任意多次请求的执行结果和一次请求的执行结果所产生的影响相同。说的直白一点就是查询操作无论查询多少次都不会影响数据本身,因此查询操作本身就是幂等的。但是新增操作,每执行一次数据库就会发生变化,所以它是非幂等的。

幂等问题的解决有很多思路,这里讲一种比较严谨的。提供一个生成随机数的接口,随机数全局唯一。调用接口的时候带入随机数。第一次调用,业务处理成功后,将随机数作为key,操作结果作为value,存入redis,同时设置过期时长。第二次调用,查询redis,如果key存在,则证明是重复提交,直接返回错误。

三 数据规范问题

3.1 版本控制

一套成熟的API文档,一旦发布是不允许随意修改接口的。这时候如果想新增或者修改接口,就需要加入版本控制,版本号可以是整数类型,也可以是浮点数类型。一般接口地址都会带上版本号,http://ip:port//v1/list。

3.2 响应状态码规范

一个牛逼的API,还需要提供简单明了的响应值,根据状态码就可以大概知道问题所在。我们采用http的状态码进行数据封装,例如200表示请求成功,4xx表示客户端错误,5xx表示服务器内部发生错误。状态码设计参考如下:

分类 描述
1xx 信息,服务器收到请求,需要请求者继续执行操作
2xx 成功
3xx 重定向,需要进一步的操作以完成请求
4xx 客户端错误,请求包含语法错误或无法完成请求
5xx 服务端错误

状态码枚举类:

public enum CodeEnum {// 根据业务需求进行添加SUCCESS(200,"处理成功"),ERROR_PATH(404,"请求地址错误"),ERROR_SERVER(505,"服务器内部发生错误");private int code;private String message;CodeEnum(int code, String message) {this.code = code;this.message = message;}public int getCode() {return code;}public void setCode(int code) {this.code = code;}public String getMessage() {return message;}public void setMessage(String message) {this.message = message;}
}

3.3 统一响应数据格式

为了方便给客户端响应,响应数据会包含三个属性,状态码(code),信息描述(message),响应数据(data)。客户端根据状态码及信息描述可快速知道接口,如果状态码返回成功,再开始处理数据。

响应结果定义及常用方法:

public class R implements Serializable {private static final long serialVersionUID = 793034041048451317L;private int code;private String message;private Object data = null;public int getCode() {return code;}public void setCode(int code) {this.code = code;}public String getMessage() {return message;}public void setMessage(String message) {this.message = message;}public Object getData() {return data;}/*** 放入响应枚举*/public R fillCode(CodeEnum codeEnum){this.setCode(codeEnum.getCode());this.setMessage(codeEnum.getMessage());return this;}/*** 放入响应码及信息*/public R fillCode(int code, String message){this.setCode(code);this.setMessage(message);return this;}/*** 处理成功,放入自定义业务数据集合*/public R fillData(Object data) {this.setCode(CodeEnum.SUCCESS.getCode());this.setMessage(CodeEnum.SUCCESS.getMessage());this.data = data;return this;}
}

总结

本篇文章从安全性、幂等性、数据规范等方面讨论了API设计规范。除此之外,一个好的API还少不了一个优秀的接口文档。接口文档的可读性非常重要,虽然很多程序员都不喜欢写文档,而且不喜欢别人不写文档。为了不增加程序员的压力,推荐使用swagger或其他接口管理工具,通过简单配置,就可以在开发中测试接口的连通性,上线后也可以生成离线文档用于管理API。

最后,再附上我历时三个月总结的 Java 面试 + Java 后端技术学习指南,笔者这几年及春招的总结,github 1.4k star,拿去不谢!下载方式1. 首先扫描下方二维码
2. 后台回复「Java面试」即可获取

如何设计一个牛逼的API接口相关推荐

  1. symfony api 错误响应_如何设计一个牛逼的 API 接口

    在日常开发中,总会接触到各种接口.前后端数据传输接口,第三方业务平台接口.一个平台的前后端数据传输接口一般都会在内网环境下通信,而且会使用安全框架,所以安全性可以得到很好的保护.这篇文章重点讨论一下提 ...

  2. ​如何设计一个安全可靠的 API 接口?

    作者 | 阿文 责编 | 屠敏 出品 | CSDN(ID:CSDNnews) 最近几年,随着RESTful API开始风靡,使用HTTP header来传递认证令牌似乎变得理所应当,通过 RESTfu ...

  3. 设计一个高质量的 API 接口

    点击关注公众号,实用技术文章及时了解 来源:blog.csdn.net/weixin_43318367/article/ details/108746057 你是否也感同身受? 对接XX业务时,XX业 ...

  4. 如何设计一个牛逼的本地缓存

    点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 来源:ksfzhaohui juejin.im/post/5dd9 ...

  5. 如何设计一个牛逼的本地缓存!

    来源:ksfzhaohui | http://dwz.win/Ws4 最近在看Mybatis的源码,刚好看到缓存这一块,Mybatis提供了一级缓存和二级缓存:一级缓存相对来说比较简单,功能比较齐全的 ...

  6. 如何设计一个牛逼的 Feed 流系统

    点击上方"肉眼品世界",选择"设为星标" 深度价值体系传递 作者:少强 简介 差不多十年前,随着功能机的淘汰和智能机的普及,互联网开始进入移动互联网时代,最具代 ...

  7. 设计一个安全对外的API接口,需要考虑哪些方面?

    点击关注公众号,实用技术文章及时了解 来源:blog.csdn.net/zhou920786312/article/ details/95536556 如何保证外网开放接口的安全性. 使用加签名方式, ...

  8. 又发现一款牛逼的 API 敏捷开发工具

    来源:xie.infoq.cn/article/b5c3a339267e1351c6151b42a   初衷 跟大家分享一个牛逼的 API 敏捷开发工具,用尽可能简单的方式,完成尽可能多的需求.通过约 ...

  9. sync神key_告诉你 Redis 是一个牛逼货

    原标题:告诉你 Redis 是一个牛逼货 概述: Redis是一个 Key-Value 存储系统.和 Memcached 类似,它支持存储的 value 类型相对更多,包括 string(字符串). ...

最新文章

  1. SAP WM模块常用T-code
  2. 1059 Prime Factors
  3. “我是技术总监,你干嘛总问我技术细节?”| 程序员有话说
  4. IOS基础基于pod上手体验FMDB框架
  5. 【无码专区6】球与盒子(数学线性筛)
  6. [剑指offer][JAVA]面试题第[20]题[表示数值的字符串][状态机]
  7. 设计模式学习之单件模式singleton
  8. 传说中的神器: shared_ptr/weak_ptr/scoped_ptr
  9. 调用阿里API获取城市天气信息
  10. 普乐郡——回乐县(城市记忆7)
  11. root 存档位置 linux,Linux 档案与目录管理(转)
  12. c语言写拼图游戏算法,[原创]拼图游戏移动算法,简单易懂
  13. Socks5代理Socks5 Proxy
  14. LenNet5-MNIST
  15. 演讲:星座决定命运(密码为作者姓名拼音)
  16. JS/JavaScript中两个等号 == 和 三个等号 === 的区别
  17. 2018清华计算机类专业录取分数线,2018年清华大学录取分数线是多少
  18. 【数据库】DBMS几种进程模型优劣与举例
  19. 网络系统集成课程设计
  20. 《影子跟随算法》目前试着自己写一个联网双人对战的小游戏 看到一篇网络同步的文章很受启发 这里记录一下

热门文章

  1. STM32学习笔记:读写内部Flash。
  2. 在机器学习分类中如何处理训练集中不平衡问题
  3. word2vec原理(一): 词向量、CBOW与Skip-Gram模型基础
  4. 初等数论--同余--WILSON定理
  5. [密码学] 强力攻击
  6. 【Opencv实战】简易版“美颜”来啦—再见旧照片,Python一键美颜哦~
  7. buu Alice与Bob
  8. 【Flask】flask入门以及第一个flask项目的创建
  9. python——常用的内置函数
  10. optee返回REE的几种方式