zoukankan      html  css  js  c++  java
  • 自动化测试如何访问不同的环境对应服务实例

    首先介绍做一下场景介绍:

      1、我们公司的测试环境比较复杂,预发环境(UAT)一套,SIT环境4套,DEV环境7套。我是负责中台模块的测试,功能类似一个订单中心,但是功能相对比较复杂。网关进来的95%以上的请求都要我负责的模块来处理(不论线上业务还是线下业务,因此所有的环境都要经过我负责模块。

         2、我们公司使用的grpc微服务框架,而我负责的中台模块,都是通过grpc的微服务接口(不提供http接口),对于测试来讲,这是个不幸的消息。

    那么我们中台的接口自动化测试是如何来实现的呢?

    这个是完整 RPC 架构图

    一个 RPC 的核心功能主要有 5 个部分组成,分别是:客户端、客户端 Stub、网络传输模块、服务端 Stub、服务端等

    • 客户端(Client):服务调用方。
    • 客户端存根(Client Stub):存放服务端地址信息,将客户端的请求参数数据信息打包成网络消息,再通过网络传输发送给服务端。
    • 服务端存根(Server Stub):接收客户端发送过来的请求消息并进行解包,然后再调用本地服务进行处理。
    • 服务端(Server):服务的真正提供者。
    • Network Service:底层传输,可以是 TCP 或 HTTP。

    了解上面的基本知识。现在来介绍我是如何实现的。

    1、创建连接到远程服务器的 channel
    2、构建使用该channel的客户端stub
    3、调用服务方法,执行RPC调用
    4、封装成Controller

     构建客户端stub

    public class Client {
    
    
        //样例 stub
        private DemoServiceGrpc.DemoServiceBlockingStub demoServiceBlockingStub;
        //原生的stub 点对点测试
        public Client() {
            ManagedChannel channel = null;
            try {
                String ip =PropertiesUtils.getValue("****.grpc.ip");
                String port = PropertiesUtils.getValue("****.grpc.port");
                channel = ManagedChannelBuilder.forTarget("static://" + ip + ":" + port).usePlaintext().build();
            } catch (Exception e) {
                e.printStackTrace();
            }
            demoServiceBlockingStub = DemoServiceGrpc.newBlockingStub(channel);
            
        }
        public DemoServiceGrpc.DemoServiceBlockingStub getdemoServiceBlockingStub() {
            return demoServiceBlockingStub;
        }
    }

     封装Controller:

     

    那么简单的http 接口服务就实现了。接下来重点来了,如何实现部署一台服务访问不同环境呢????

    具体实现:

    基于spring提供原生的 AbstractRoutingDataSource ,参考一些文档自己实现切换

    1、 为了区分不同环境的配置,采用了application-{}.yaml文件来隔离, 然后通过application.yaml文件来控制加载所有的配置文件

    2、application.yaml配置

    3、在Springboot的启动类上,排除掉datasource自动配置

    @SpringBootApplication(exclude = {DataSourceAutoConfiguration.class})
    public class GrpcApplication {
    
        public static void main(String[] args) {
            SpringApplication.run(GrpcApplication.class, args);
        }
    
    }

    4、新建一个EnvContext类,采用ThreadLocal的方式,对每个请求线程的环境变量进行隔离,这里容易遇到坑,springboot都是内嵌的tomcat启动模式,如果tomcat设置了链接的重用规则,那么如果env的信息没有被清除,可能会导致错误加载配置

    /**
     * 用来存放环境的变量,用于动态的去切换
     */
    public class EnvContext {
     
        public static ThreadLocal<String> envThreadLocal = new InheritableThreadLocal<>();
     
        public static String getEnv(){
            return envThreadLocal.get();
        }
     
        public static void setEnv(String env){
            envThreadLocal.set(env);
        }
     
        public static void clear(){
            envThreadLocal.remove();
        }
    }

    5、创建一个DynamicDataSource, 这里继承了AbstractRoutingDataSource,动态数据源类集成了Spring提供的AbstractRoutingDataSource类,AbstractRoutingDataSource 中获取数据源的方法就是 determineTargetDataSource,而此方法又通过 determineCurrentLookupKey 方法获取查询数据源的key。

    public class DynamicDataSource extends AbstractRoutingDataSource {
     
     
        @Override
        protected Object determineCurrentLookupKey() {
            return EnvContext.getEnv();
        }
    }

    6、定义一个枚举类,放入所有的环境信息

    @Getter
    @AllArgsConstructor
    @NoArgsConstructor
    public enum EnvEnum {
    
        DEV1("dev1","开发环境dev1"),
        DEV2("dev2","开发环境dev2"),
        DEV3("dev3","开发环境dev3"),
        DEV4("dev4","开发环境dev4"),
        DEV5("dev5","开发环境dev5"),
        DEV6("dev6","开发环境dev6"),
        DEV7("dev7","开发环境dev7"),
        SIT1("sit1","集成环境SIT1"),
        SIT2("sit2","集成环境SIT2"),
        SIT3("sit3","集成环境SIT3"),
        SIT4("sit4","集成环境SIT4"),
        UAT("uat","集成环境UAT");
        public String env;
        public String desc;
    }

    7、重点来了,我们通过AOP, 去拿到每次http的请求头中的header信息,来动态的切换EnvContext中的env配置

    @Aspect
    @Component
    @Slf4j
    public class EnvAop {
    
        public ThreadLocal<String> threadLocal = new ThreadLocal<>();
    
        @Pointcut("@annotation(org.springframework.web.bind.annotation.RequestMapping ) && @annotation(io.swagger.annotations.ApiOperation))")
        public void ex(){}
    
    
        @Around("ex()")
        public Object envAop(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
            Object result;
            try {
                //获取每个请求的header,拿到环境变量的参数,存入ThreadLocal中,供每个线程使用
                RequestAttributes ra = RequestContextHolder.getRequestAttributes();
                ServletRequestAttributes sra = (ServletRequestAttributes) ra;
                HttpServletRequest request = sra.getRequest();
                // 获取请求头
                Enumeration<String> enumeration = request.getHeaderNames();
                //
                String env = request.getHeader("env");
                if(StringUtils.isEmpty(env)){
                    log.info("~~~~ 拦截到http请求,环境变量信息为空,设置为默认dev1", env);
                    env = EnvEnum.DEV1.env;
                }
                log.info("~~~~ 拦截到http请求,环境变量信息为{}", env);
                EnvConfig.envThreadLocal.set(env);
                result = proceedingJoinPoint.proceed();
    
            } finally {
                //请求结束后,将环境变量的信息从ThreadLocal中移除
                EnvConfig.clear();
                log.info("~~~~ http请求结束,重置env的信息为{}" , EnvConfig.getEnv());
            }
            return result;
        }
    }

    8、编写一个工具类,动态获取Spring的容器ApplicationContext

    @Component
    public class SpringContextUtil {
    
        @Resource
        private ApplicationContext applicationContext;
    
        private static ConfigurableApplicationContext context;
        private static BeanFactory factoryBean;
    
        @PostConstruct
        public void init() {
            context = (ConfigurableApplicationContext) applicationContext;
            factoryBean = context.getBeanFactory();
        }
    
        public static BeanFactory getFactoryBean() {
            return factoryBean;
        }
    
        public static ConfigurableApplicationContext getApplicationContext() {
            return context;
        }
    }

    9、然后编写一个配置信息动态读取工具类,每次请求进来,env会动态切换,然后工具类会自动拼装env信息去读取

      

    public class PropertiesUtils {
        public static String getValue(String key) throws Exception {
            Environment environment = (Environment) SpringContextUtil.getApplicationContext().getBean("environment");
            String value = environment.getProperty(EnvConfig.getEnv() + "." + key);
            if(StringUtils.isEmpty(value)){
                throw new Exception("配置信息获取失败,请检查application-"+ EnvConfig.getEnv()+".yaml文件!, key = " + key + " , env = " + EnvConfig.getEnv());
            }
            return value;
        }
    }

    到此  实现通过http请求 中header中配置env参数来实现  动态切换服务器(以此类推可以修改同过parame或者url中的参数来实现动态切换服务器)

     重点注意:

       实现client 的连接的方法不能通过Springboot  的@service @Autowired来实现  不然无法实现动态切换服务器  也就是Controller里面每次使用client的时候  都要new

            因为通过Bean实现的话,启动的时候就已经加载完成了,无法实现动态加载

     

    声明:该文章参考公司同事(章帅)的文章

     

      

  • 相关阅读:
    java实现第四届蓝桥杯核桃的数量
    java实现第四届蓝桥杯核桃的数量
    java实现第四届蓝桥杯核桃的数量
    java实现第四届蓝桥杯核桃的数量
    java实现第四届蓝桥杯逆波兰表达式
    jsp中生成的验证码和存在session里面的验证码不一致的处理
    使用session实现一次性验证码
    java图形验证码生成工具类及web页面校验验证码
    关于在页面得到的servlet验证码总是上一次保存在session中的
    FreeMarker 获取页面request、session
  • 原文地址:https://www.cnblogs.com/emars/p/12750492.html
Copyright © 2011-2022 走看看