手机版
位置:筑能财经 > 社区 >

每日热点:[云原生]接口安全方案提供和实践

来源:腾讯云 | 2023-03-31 14:20:19

为什么要保证接口安全

对于互联网来说,只要你系统的接口暴露在外网,就避免不了接口安全问题。 如果你的接口在外网裸奔,只要让黑客知道接口的地址和参数就可以调用,那简直就是灾难。

举个例子:你的网站用户注册的时候,需要填写手机号,发送手机验证码,如果这个发送验证码的接口没有经过特殊安全处理,那这个短信接口早就被人盗刷不知道浪费多少钱了。

那如何保证接口安全呢?


(资料图片)

一般来说,暴露在外网的api接口需要做到防篡改防重放才能称之为安全的接口。

防篡改

我们知道http 是一种无状态的协议,服务端并不知道客户端发送的请求是否合法,也并不知道请求中的参数是否正确。

举个例子, 现在有个充值的接口,调用后可以给用户增加对应的余额。

http://localhost/api/user/recharge?user_id=1001&amount=10

如果非法用户通过抓包获取到接口参数后,修改user_id 或 amount的值就可以实现给任意账户添加余额的目的。

1,如何解决

采用https协议可以将传输的明文进行加密,但是黑客仍然可以截获传输的数据包,进一步伪造请求进行重放攻击。如果黑客使用特殊手段让请求方设备使用了伪造的证书进行通信,那么https加密的内容也会被解密。

一般的做法有2种:

采用https方式把接口的数据进行加密传输,即便是被黑客破解,黑客也花费大量的时间和精力去破解。接口后台对接口的请求参数进行验证,防止被黑客篡改;步骤1:客户端使用约定好的秘钥对传输的参数进行加密,得到签名值sign1,并且将签名值也放入请求的参数中,发送请求给服务端步骤2:服务端接收到客户端的请求,然后使用约定好的秘钥对请求的参数再次进行签名,得到签名值sign2。步骤3:服务端比对sign1和sign2的值,如果不一致,就认定为被篡改,非法请求。

防重放

防重放也叫防复用。简单来说就是我获取到这个请求的信息之后什么也不改,,直接拿着接口的参数去 重复请求这个充值的接口。此时我的请求是合法的, 因为所有参数都是跟合法请求一模一样的。重放攻击会造成两种后果:

针对插入数据库接口:重放攻击,会出现大量重复数据,甚至垃圾数据会把数据库撑爆。针对查询的接口:黑客一般是重点攻击慢查询接口,例如一个慢查询接口1s,只要黑客发起重放攻击,就必然造成系统被拖垮,数据库查询被阻塞死。

对于重放攻击一般有两种做法:

基于timestamp的方案

每次HTTP请求,都需要加上timestamp参数,然后把timestamp和其他参数一起进行数字签名。因为一次正常的HTTP请求,从发出到达服务器一般都不会超过60s,所以服务器收到HTTP请求之后,首先判断时间戳参数与当前时间比较,是否超过了60s,如果超过了则认为是非法请求。

一般情况下,黑客从抓包重放请求耗时远远超过了60s,所以此时请求中的timestamp参数已经失效了。 如果黑客修改timestamp参数为当前的时间戳,则sign1参数对应的数字签名就会失效,因为黑客不知道签名秘钥,没有办法生成新的数字签名。

但是这种方式的漏洞也是显而易见,如果在60s之内进行重放攻击,那就没办法了,所以这种方式不能保证请求仅一次有效。

老鸟们一般会采取下面这种方案,既可以解决接口重放问题,又可以解决一次请求有效的问题。

基于nonce + timestamp 的方案

nonce的意思是仅一次有效的随机字符串,要求每次请求时该参数要保证不同。实际使用用户信息+时间戳+随机数等信息做个哈希之后,作为nonce参数。

此时服务端的处理流程如下:

去 redis 中查找是否有 key 为 nonce:{nonce}的 string如果没有,则创建这个 key,把这个 key 失效的时间和验证 timestamp 失效的时间一致,比如是 60s。如果有,说明这个 key 在 60s 内已经被使用了,那么这个请求就可以判断为重放请求。

这种方案nonce和timestamp参数都作为签名的一部分传到后端,基于timestamp方案可以让黑客只能在60s内进行重放攻击,加上nonce随机数以后可以保证接口只能被调用一次,可以很好的解决重放攻击问题。

代码实现

接下来以SpringBoot项目为例看看如何实现接口的防篡改和防重放功能。

1、构建请求头对象

@Data@Builderpublic class RequestHeader {   private String sign ;   private Long timestamp ;   private String nonce;}

2、工具类从HttpServletRequest获取请求参数

@Slf4j@UtilityClasspublic class HttpDataUtil {    /**     * post请求处理:获取 Body 参数,转换为SortedMap     *     * @param request     */    public  SortedMap getBodyParams(final HttpServletRequest request) throws IOException {        byte[] requestBody = StreamUtils.copyToByteArray(request.getInputStream());        String body = new String(requestBody);        return JsonUtil.json2Object(body, SortedMap.class);    }​​    /**     * get请求处理:将URL请求参数转换成SortedMap     */    public static SortedMap getUrlParams(HttpServletRequest request) {        String param = "";        SortedMap result = new TreeMap<>();​        if (StringUtils.isEmpty(request.getQueryString())) {            return result;        }​        try {            param = URLDecoder.decode(request.getQueryString(), "utf-8");        } catch (UnsupportedEncodingException e) {            e.printStackTrace();        }​        String[] params = param.split("&");        for (String s : params) {            String[] array=s.split("=");            result.put(array[0], array[1]);        }        return result;    }}

这里的参数放入SortedMap中对其进行字典排序,前端构建签名时同样需要对参数进行字典排序。

3、签名验证工具类

@Slf4j@UtilityClasspublic class SignUtil {    /**     * 验证签名     * 验证算法:把timestamp + JsonUtil.object2Json(SortedMap)合成字符串,然后MD5     */    @SneakyThrows    public  boolean verifySign(SortedMap map, RequestHeader requestHeader) {        String params = requestHeader.getNonce() + requestHeader.getTimestamp() + JsonUtil.object2Json(map);        return verifySign(params, requestHeader);    }​    /**     * 验证签名     */    public boolean verifySign(String params, RequestHeader requestHeader) {        log.debug("客户端签名: {}", requestHeader.getSign());        if (StringUtils.isEmpty(params)) {            return false;        }        log.info("客户端上传内容: {}", params);        String paramsSign = DigestUtils.md5DigestAsHex(params.getBytes()).toUpperCase();        log.info("客户端上传内容加密后的签名结果: {}", paramsSign);        return requestHeader.getSign().equals(paramsSign);    }}

4、HttpServletRequest包装类

public class SignRequestWrapper extends HttpServletRequestWrapper {    //用于将流保存下来    private byte[] requestBody = null;​    public SignRequestWrapper(HttpServletRequest request) throws IOException {        super(request);        requestBody = StreamUtils.copyToByteArray(request.getInputStream());    }​    @Override    public ServletInputStream getInputStream() throws IOException {        final ByteArrayInputStream bais = new ByteArrayInputStream(requestBody);​        return new ServletInputStream() {            @Override            public boolean isFinished() {                return false;            }​            @Override            public boolean isReady() {                return false;            }​            @Override            public void setReadListener(ReadListener readListener) {​            }​            @Override            public int read() throws IOException {                return bais.read();            }        };​    }​    @Override    public BufferedReader getReader() throws IOException {        return new BufferedReader(new InputStreamReader(getInputStream()));    }}

防篡改和防重放我们会通过SpringBoot Filter来实现,而编写的filter过滤器需要读取request数据流,但是request数据流只能读取一次,需要自己实现HttpServletRequestWrapper对数据流包装,目的是将request流保存下来。

5、创建过滤器实现安全校验

@Configurationpublic class SignFilterConfiguration {    @Value("${sign.maxTime}")    private String signMaxTime;​    //filter中的初始化参数    private Map initParametersMap =  new HashMap<>();​    @Bean    public FilterRegistrationBean contextFilterRegistrationBean() {        initParametersMap.put("signMaxTime",signMaxTime);        FilterRegistrationBean registration = new FilterRegistrationBean();        registration.setFilter(signFilter());        registration.setInitParameters(initParametersMap);        registration.addUrlPatterns("/sign/*");        registration.setName("SignFilter");        // 设置过滤器被调用的顺序        registration.setOrder(1);        return registration;    }​    @Bean    public Filter signFilter() {        return new SignFilter();    }}
@Slf4jpublic class SignFilter implements Filter {    @Resource    private RedisUtil redisUtil;​    //从fitler配置中获取sign过期时间    private Long signMaxTime;​    private static final String NONCE_KEY = "x-nonce-";​    @Override    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {        HttpServletRequest httpRequest = (HttpServletRequest) servletRequest;        HttpServletResponse httpResponse = (HttpServletResponse) servletResponse;​        log.info("过滤URL:{}", httpRequest.getRequestURI());​        HttpServletRequestWrapper requestWrapper = new SignRequestWrapper(httpRequest);        //构建请求头        RequestHeader requestHeader = RequestHeader.builder()                .nonce(httpRequest.getHeader("x-Nonce"))                .timestamp(Long.parseLong(httpRequest.getHeader("X-Time")))                .sign(httpRequest.getHeader("X-Sign"))                .build();​        //验证请求头是否存在        if(StringUtils.isEmpty(requestHeader.getSign()) || ObjectUtils.isEmpty(requestHeader.getTimestamp()) || StringUtils.isEmpty(requestHeader.getNonce())){            responseFail(httpResponse, ReturnCode.ILLEGAL_HEADER);            return;        }​        /*         * 1.重放验证         * 判断timestamp时间戳与当前时间是否操过60s(过期时间根据业务情况设置),如果超过了就提示签名过期。         */        long now = System.currentTimeMillis() / 1000;​        if (now - requestHeader.getTimestamp() > signMaxTime) {            responseFail(httpResponse,ReturnCode.REPLAY_ERROR);            return;        }​        //2. 判断nonce        boolean nonceExists = redisUtil.hasKey(NONCE_KEY + requestHeader.getNonce());        if(nonceExists){            //请求重复            responseFail(httpResponse,ReturnCode.REPLAY_ERROR);            return;        }else {            redisUtil.set(NONCE_KEY+requestHeader.getNonce(), requestHeader.getNonce(), signMaxTime);        }​​        boolean accept;        SortedMap paramMap;        switch (httpRequest.getMethod()){            case "GET":                paramMap = HttpDataUtil.getUrlParams(requestWrapper);                accept = SignUtil.verifySign(paramMap, requestHeader);                break;            case "POST":                paramMap = HttpDataUtil.getBodyParams(requestWrapper);                accept = SignUtil.verifySign(paramMap, requestHeader);                break;            default:                accept = true;                break;        }        if (accept) {            filterChain.doFilter(requestWrapper, servletResponse);        } else {            responseFail(httpResponse,ReturnCode.ARGUMENT_ERROR);            return;        }​    }​    private void responseFail(HttpServletResponse httpResponse, ReturnCode returnCode)  {        ResultData resultData = ResultData.fail(returnCode.getCode(), returnCode.getMessage());        WebUtils.writeJson(httpResponse,resultData);    }​    @Override    public void init(FilterConfig filterConfig) throws ServletException {        String signTime = filterConfig.getInitParameter("signMaxTime");        signMaxTime = Long.parseLong(signTime);    }}

6、Redis工具类

@Componentpublic class RedisUtil {    @Resource    private RedisTemplate redisTemplate;​    /**     * 判断key是否存在     * @param key 键     * @return true 存在 false不存在     */    public boolean hasKey(String key) {        try {            return Boolean.TRUE.equals(redisTemplate.hasKey(key));        } catch (Exception e) {            e.printStackTrace();            return false;        }    }​​    /**     * 普通缓存放入并设置时间     * @param key   键     * @param value 值     * @param time  时间(秒) time要大于0 如果time小于等于0 将设置无限期     * @return true成功 false 失败     */    public boolean set(String key, Object value, long time) {        try {            if (time > 0) {                redisTemplate.opsForValue().set(key, value, time, TimeUnit.SECONDS);            } else {                set(key, value);            }            return true;        } catch (Exception e) {            e.printStackTrace();            return false;        }    }​    /**     * 普通缓存放入     * @param key   键     * @param value 值     * @return true成功 false失败     */    public boolean set(String key, Object value) {        try {            redisTemplate.opsForValue().set(key, value);            return true;        } catch (Exception e) {            e.printStackTrace();            return false;        }    }​}

标签:

上一篇:

下一篇:

精彩放送

新闻资讯

  1. 周三银银间回购定盘利率涨跌互现(03-29)
  1. 天天热点评!中国教育考试网四级登录入口
  1. 要闻速递:黑龙江停息挂账有哪些方式?停息挂账容易申请吗?
  1. 环球最资讯丨特斯拉新款Model S/X开启中国市场交付
  1. 每日视点!北京初中学考第二次英语听说机考本周日举行
  1. 世界热讯:加速向新能源转型:启辰DD-i超混动技术将与比亚迪DM-i一战?
  1. 全球观热点:cad基础教程(cad制图初学入门教程有哪些)
  1. 每日速递:为避开欧盟反垄断调查 微软据称提出改变云计算策略
  1. 宠物行业能否成为朝云集团新增长曲线
  1. 近四成QDII月内实现正收益 黄金、科技等主题表现亮眼
  1. 国家线什么时候出(什么时候出国家线)
  1. 沙漠同样是主场 杜卡迪Desert X初体验
  1. 金猫银猫(01815)公布2022年业绩 公司拥有人应占亏损约2560万元 同比盈转亏
  1. 【世界新要闻】露露乐蒙四季度净营收27.7亿美元 分析师预期27.0亿美元
  1. 每日热文:歌手李娜:不顾亲人流泪劝阻出家,25年后,她过得怎样了?
  1. 世态炎凉不过为人走茶凉什么意思_人走茶凉什么意思
  1. 播报:【零基础学经济Ep167】高鸿业《西方经济学(宏观)》概念梳理P601:新古典宏观经济学
  1. 全球热讯:40家企业专场路演,河南加速推动优质资本与项目对接
  1. 她是娱乐圈低调女演员,潘长江竟是她老爸,如今嫁入豪门家庭幸福
  1. 罕见!头部券商收到警示函,原因竟是这个…
  1. 龙井茶味、荷叶香味、桂花味……这些口味的啤酒你喝过吗?杭州西湖推出“西湖啤酒·醉西湖”系列啤酒
  1. 当前热讯:三国志战略版吴国肉弓阵容推荐 三国志战略版吴国肉弓阵容推荐最新
  1. 当前讯息:引进企业336家!保定“类中关村”服务育出 “金”种子
  1. 东华大学Z世代设计师集体亮相上海时装周,新潮作品表达时尚新思考
  1. 环球热文:红绿灯故障,开车可以直接闯过去吗?交警:这样做才不会被罚
  1. 确定阳虚的人怎么调理好?
  1. 精彩看点:章丘区:打造“百脉汇”人才品牌 构建青年人才集聚强磁场
  1. 安徽出台106项举措提升创优营商环境
  1. 每日速读!小奖不断大奖难觅 抽奖游戏“一番赏”销售存猫腻
  1. 韩岗“韭”香飘全国
  1. 全球信息:河堤筑起“安全屏风”
  1. 深夜突袭抓获42人,昆明经开警方捣毁隐藏深山流动赌博窝点
  1. 天天热推荐:3月28日毫米波概念板块跌幅达2%
  1. 全球热议:消防安全知识培训内容ppt(消防安全知识培训内容)
  1. 今日视点:上饶广丰区:“以花造节” 打造城市新名片
  1. 环球聚焦:青春无敌!武汉盛帆女篮大胜北京首钢获U19季军 三大新星闪耀
  1. 黑暗料理王!曼城王牌最爱中餐 五样菜混合大搅拌
  1. 【全球快播报】模拟基金(600980)
  1. 小雷:只要列维掌权热刺就无法夺冠,球队从不以夺冠为目标
  1. 临高开展“全国中小学生安全教育日”活动
  1. 诺德基金朱明睿:医药估值已处于相对低位,看好2023年医药行情
  1. 环球看点!养老FOF业绩分化加剧 十二只产品成立以来跌逾一成
  1. 环球动态:TMT成交占比创新高 是行情见顶的信号吗?
  1. 世界观速讯丨江西省应急厅召开全省矿山安全生产工作会议
  1. 暴涨65%!“当红炸子鸡”同花顺股东拟集体减持 或套现超88亿
  1. 天天报道:莲花健康看上自嗨锅
  1. 世界即时:逾四成QDII基金近一月实现正收益 7只黄金主题产品包揽榜单前七
  1. 世界热资讯!8家公募基金公司去年经营数据出炉 营收净利增幅冷暖不均
  1. 环球热文:从几秒,到30分钟,你只差这一步,老中医今天带你跨过男人门槛!
  1. 什么叫全开麦半开麦_全开麦半开麦介绍
  1. 7万股东沸腾!中东土豪扫货A股 千亿龙头迎重磅利好!英国发生“重大事件”
  1. 仰天长啸出门去我辈岂是蓬蒿人讲的是李白背后_仰天长啸出门去我辈岂是蓬蒿人
  1. 天天要闻:孔蒂告别热刺:深深感谢俱乐部的每个人,祝愿你们未来一切顺利
  1. 年轻人收破烂、摆地摊,可不是励志故事
  1. 怎么养牛蛙蝌蚪_这两点要注意
  1. 【天天播资讯】厦门今明雨水暂歇,后天又将出现降水天气
  1. 【播资讯】Linux安装elasticsearch-head
  1. 世界视点!四川遂宁:4月起住房公积金可又贷又取,房屋套数认贷不认房
  1. 环球热点评!丰元股份: 中泰证券股份有限公司关于山东丰元化学股份有限公司2022年度保荐工作报告
  1. 每日聚焦:【小里帮忙】楼上爆管,装修一半的新房被泡一天一夜,成了“水洗房”
  1. 每日头条!宝兰德:股东易存道、易东兴拟合计减持不超4.61%
  1. 世界热门:iqc是什么意思_怎么理解iqc的意思
  1. 人民财评:让绿色建材走进千家万户还需加倍努力
  1. 快资讯:我国掺杂系列碳化硅纤维产业化应用研讨会在京召开
  1. 焦作国保档案|第2集 邘国古城
  1. 华安基金:避险情绪与加息降温主导,黄金显著净流入
  1. 嘉实碳中和主题基金3月29日提前结募
  1. 南山荟芳园房子怎么样_南山荟芳园
  1. 巴萨网红球迷:梅西哥哥告诉我希望梅西能尽快回巴萨
  1. 【世界快播报】鲜艳色≠廉价、俗气,奶奶们穿“花”的4个技巧,靓丽显年轻
  1. 环球即时:甘特图:作为管理者,先做好这件事
  1. 世界今头条!国寿安保基金:权益市场预期波动或是常态
  1. 保利发展125亿元定增遭上交所问询:要求说明14个募投房地产项目是否属于“保交楼”等
  1. 环球最新:技于指尖俯仰千年
  1. 【天天报资讯】海口发布冰雹橙色预警和雷雨大风黄色预警
  1. 每日焦点!泰州市流动就业养老金从哪儿领钱的
  1. 焦点热文:REITs新一轮扩容开启 产业园区REITs表现亮眼
  1. 微资讯!曙光乍现 传媒基金“漂”回本位
  1. 全球快资讯:TikTok 欲加之罪
  1. 焦点热门:知名基金经理“隐形重仓股”曝光 医药等行业受关注
  1. 踏空信创失之东隅 “长期主义”收之桑榆
  1. 天天看点:中欧基金:主题行情下市场波动显著加大
  1. 环球快讯:今年来平均回报超44% 游戏主题ETF“霸榜”
  1. 视焦点讯!资金逆势涌入 新能车主题ETF份额年内增逾30%
  1. 友山基金24只基金年内7只下跌 2只累计亏损超20%
  1. “迷你基金”遭遇“紧箍咒” 公募保壳上演“猫鼠游戏”
  1. 即时焦点:江南style歌词什么意思_歌词含义介绍
  1. 金价攀升难解规模之痛 部分黄金ETF倒在黎明前
  1. 今头条!钢银电商:全国钢市库存环比减少56.35万吨
  1. 房屋知识科普:loft公寓房产证写办公性质
  1. 快看点丨因为它 达尔文进化论最大的漏洞:有救了!
  1. 每日动态!肛门外有个小肉疙瘩图_肛门外有个小肉疙瘩痛是怎么回事
  1. 每日看点!二十年,抚不平伊拉克人撕心裂肺的痛
  1. 观热点:时隔1152天 上海国际邮轮母港恢复常态化运营
  1. 天天快消息!王文涛部长会见宝马集团董事长齐普策
  1. 世界速看:拉巴
  1. 省轮滑冰球邀请赛圆满落幕 成都风云、JOKER摘桂
  1. 世界热推荐:剂型分类种类_剂型
  1. 上海市中心有个奇怪的商场,靠近人民广场,就在地铁口,却没人气
  1. 世界视点!梦回唐朝佟丽娅大结局