毕设之-Hlang后端架构-双系统交互

文章目录

前言

前天我们完成了基本的整合,但是还没有整合到我们的业务系统,也就是博客系统。本来昨天要搞一手的,但是在练车,所以就没有完成。那么今天补上,差不多20号去上班实习,这点时间赶紧搞完。

首先的话,我们这里建好了业务表:

并且我们创建好了新的服务:

并且,我们通过人人开源生成好了对应的后端的一个CURD和基本接口。

然后,我们把对应的controller,和feign接口做好对接。

如图:(整合好的feign组件)

然后在我们的人人开源中台系统做好调用

这里的话,我们可以发现是这样的流程

之后的话,在我们的博客系统当中我们有这个:

交互流程

那么这里,我们今天要实现的就是在人人开源中台系统过来调用到我们的具体服务的api的时候要做到的一个权限验证

因为我们这是两个系统,没有共同使用一个用户表,这也就是意味着,我们一个权限不方便公用一个。所以对应的一些服务得像第三方api一样提供。只不过这个api接口,可以通过注册中心知道,然后可以由open-feign调用而已。

基本流程

okey,那么在这里的话,我们先简单梳理一下,我们的基本流程:

那么在这里,我们可以解读到这几个信息:

  1. 需要一个公钥接口,拿到公钥
  2. 需要一个申请临时密钥的接口,通过这个密钥去访问服务

那么在人人中台要做:

  1. 先拿到公钥
  2. 通过公钥拿到私钥
  3. 携带私钥去访问

对于博客系统:

  1. 开发公钥接口
  2. 开发私钥接口
  3. 对受保护接口进行私钥验证

同时,由于公钥接口是公开的,得到的公钥在一定时间是一致的,因此我们可以直接做个约定,公钥是啥,也就是可以学一下rouyi,这个写死省下一个接口

所以这里的话,我们在编码阶段,我可以这样干。

约定公钥

我们先约定公钥。

我们可以看到,在我们的博客系统当中的话,有个获取私钥的接口:

这里的话,我们写死了,就是直接指定公钥是HUTEROX。

人人中台携带公钥获取私钥

然后就是人人中台携带公钥获取私钥

这里的话,我们在feign的拦截器当中实现了这个功能。我们先拿到私钥。

私钥生成

那么这个时候,我们自然就要生成私钥了,这个私钥由我们的博客系统生成。

java 复制代码
package com.huterox.hlangserver.controller.innocontroller.auto;

import com.huterox.common.utils.R;

/**
 * 负责实现对内部服务调用的一个API授权签发
 * */
public interface AutoInnoApiService {

    /**
     * 签发临时授权码
     * */
    R SignApiAutoCode();

    /**
     * 验证签发的授权码是否通过
     * */
    boolean AcceptApiAutoCode(String signKey,String signal);
}

之后的话就是我们具体的实现类了。

java 复制代码
package com.huterox.hlangserver.controller.innocontroller.auto.impl;
import com.alibaba.fastjson.JSON;
import com.huterox.common.sysEntity.SignApiCode;
import com.huterox.common.utils.CodeUtils;
import com.huterox.common.utils.R;
import com.huterox.hlangserver.controller.innocontroller.auto.AutoInnoApiService;
import com.huterox.hlangserver.utils.RedisTransKey;
import com.huterox.hlangserver.utils.RedisUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.concurrent.TimeUnit;

@Service
public class BaseAutoInnoApiServiceImpl implements AutoInnoApiService {

    private RedisUtils redisUtils;

    public BaseAutoInnoApiServiceImpl(RedisUtils redisUtils) {
        this.redisUtils = redisUtils;
    }

    @Autowired
    public void setRedisUtils(RedisUtils redisUtils) {
        this.redisUtils = redisUtils;
    }

    @Override
    public R SignApiAutoCode() {
        //生成验证Sign
        String signKey = CodeUtils.creatCode(10);
        String signal = CodeUtils.creatCode(20);
        SignApiCode signApiCode = new SignApiCode();
        signApiCode.setSignal(signal);
        signApiCode.setSignKey(signKey);
        //设置30秒过期
        redisUtils.set(RedisTransKey.setServerSysKey(signKey),
                signApiCode,30, TimeUnit.SECONDS
                );
        return R.ok().put("sign",signApiCode);
    }

    @Override
    public boolean AcceptApiAutoCode(String signKey,String signal) {
        //验证传回来的签发对不对,然后把这个进行删除
        Object o = redisUtils.get(RedisTransKey.getServerSysKey(signKey));
        SignApiCode signApiCode = JSON.parseObject(o.toString(), SignApiCode.class);
        redisUtils.del(RedisTransKey.getServerSysKey(signKey));
        return signApiCode.getSignal().equals(signal)&&signApiCode.getSignKey().equals(signKey);
    }
}

在这里的话,我用了几个工具类。当然这个在先前的whitehole都提到过。这里的话就不复述了,百度都能搞定。当然重要的是,这个项目后期也是开源的,这里只是记录一下具体的做法而已。后期可能会对代码进行优化,但是基本流程是不会发生太大改变的。

人人中台携带私钥访问

之后的话,就是我们的人人中台去携带我们的私钥去访问

java 复制代码
package io.renren.config;
import com.huterox.common.sysEntity.SignApiCode;
import com.huterox.common.utils.FastJsonUtils;
import com.huterox.common.utils.R;
import com.huterox.feign.server.FeignAutoInnoApiSeverService;
import feign.RequestInterceptor;
import feign.RequestTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;

import java.util.HashMap;
import java.util.Map;

@Configuration
public class FeignRequestInterceptor implements RequestInterceptor {

    private FeignAutoInnoApiSeverService feignAutoInnoApiSeverService;

    public FeignRequestInterceptor(FeignAutoInnoApiSeverService feignAutoInnoApiSeverService) {
        this.feignAutoInnoApiSeverService = feignAutoInnoApiSeverService;
    }

    @Autowired
    public void setFeignAutoInnoApiSeverService(FeignAutoInnoApiSeverService feignAutoInnoApiSeverService) {
        this.feignAutoInnoApiSeverService = feignAutoInnoApiSeverService;
    }

    @Override
    public void apply(RequestTemplate template) {

        //先那到临时签证
        Map<String, String> map = new HashMap<>();
        map.put("innoHeader","HUTEROX");
        R sign = feignAutoInnoApiSeverService.getSign(map);

        //拿到签证去正式访问保护接口
        String signKey = "";
        String signal = "";
        int code = (int) sign.get("code");
        if(code==0){
            String sign1 = FastJsonUtils.toJson(sign.get("sign"));
            SignApiCode signApiCode = FastJsonUtils.fromJson(sign1, SignApiCode.class);
            signKey = signApiCode.getSignKey();
            signal = signApiCode.getSignal();
        }
        //发送远程调用请求,填写signApiCode
        template.header("signKey",signKey);
        template.header("signal",signal);
    }

}

私钥验证(博客系统)

之后的话,我们的博客系统就要去验证我们的密钥对不对。(其实你发现,这玩意不就是服务之间的验证码嘛)

这里的话,我们受保护的接口是这些:

所以的话,我们来创建一个切面来进行处理:

java 复制代码
package com.huterox.hlangserver.aspect;
import com.huterox.common.utils.R;
import com.huterox.hlangserver.controller.innocontroller.auto.AutoInnoApiService;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;

import javax.servlet.http.HttpServletRequest;

@Aspect
@Component
public class InnoAutoAspect {

    @Pointcut("within(com.huterox.hlangserver.controller.innocontroller.api..*) && @annotation(org.springframework.web.bind.annotation.RequestMapping)")
    public void verification() {}

    private AutoInnoApiService autoInnoApiService;

    public InnoAutoAspect(AutoInnoApiService autoInnoApiService) {
        this.autoInnoApiService = autoInnoApiService;
    }

    @Autowired
    public void setAutoInnoApiService(AutoInnoApiService autoInnoApiService) {
        this.autoInnoApiService = autoInnoApiService;
    }

    @Around("verification()")
    public R verification(ProceedingJoinPoint joinPoint) throws Throwable {
        HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest();
        String signKey = request.getHeader("signKey");
        String signal = request.getHeader("signal");
        if(!autoInnoApiService.AcceptApiAutoCode(signKey, signal)){
            return R.error();
        }
        return (R) joinPoint.proceed();
    }
}

调试演示

okey,那么之后的话,我们来演示具体流程

(完整的前端怎么整合的我就不演示了,以前的博客都有,或者百度都可以)

我们来看到这个:

拿到私钥,然后扩充访问

之后的话再我们那个博客系统可以看到方行。

总结

okey,以上的就是我们今天的内容,当然这部分还有优化的点,首先就是我们的人人中台,在访问的时候,我们用的是apply,重写了里面的拦截。但是问题在于,这个是全局通用的,如果我们后面有另一个子系统,那么这个时候的话,就建议使用切面了。当然这里这样处理是因为,这里我就只设计两个子系统,上次的whitehole有9个微服务,服务器顶不住。所以这次不乱搞了,省点资源,后面再研究研究这个网站智能助手玩玩。

(ps: 有没有广州的大哥春招再联系联系(狗头))

相关推荐
盛派网络小助手3 小时前
微信 SDK 更新 Sample,NCF 文档和模板更新,更多更新日志,欢迎解锁
开发语言·人工智能·后端·架构·c#
快乐非自愿7 小时前
分布式系统架构2:服务发现
架构·服务发现
2401_854391087 小时前
SSM 架构中 JAVA 网络直播带货查询系统设计与 JSP 有效实现方法
java·开发语言·架构
264玫瑰资源库7 小时前
从零开始C++棋牌游戏开发之第二篇:初识 C++ 游戏开发的基本架构
开发语言·c++·架构
神一样的老师7 小时前
面向高精度网络的时间同步安全管理架构
网络·安全·架构
2401_857026237 小时前
基于 SSM 架构的 JAVA 网络直播带货查询系统设计与 JSP 实践成果
java·开发语言·架构
9527华安7 小时前
FPGA实现MIPI转FPD-Link车载同轴视频传输方案,基于IMX327+FPD953架构,提供工程源码和技术支持
fpga开发·架构·mipi·imx327·fpd-link·fpd953
DT辰白7 小时前
如何解决基于 Redis 的网关鉴权导致的 RESTful API 拦截问题?
后端·微服务·架构
264玫瑰资源库8 小时前
从零开始C++棋牌游戏开发之第四篇:牌桌界面与交互实现
开发语言·c++·交互
温轻舟8 小时前
前端开发 之 12个鼠标交互特效上【附完整源码】
开发语言·前端·javascript·css·html·交互·温轻舟