基于jvm-sandbox的imock开发指南详解

 更新时间:2023年08月11日 09:06:42   作者:李梨同学  
这篇文章主要为大家介绍了基于jvm-sandbox的imock开发指南详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

需求

  • 支持java后端服务方法级别的mock,对没有测试环境的第三方服务进行mock,增加团队覆盖率。
  • 启用,返回用户自定义的mock响应结果
  • 停用,返回服务本身的结果

通过一系列调研,最终工具选型了基于 jvm-sandbox的mock服务,是Java方法级别的mock,操作就是监听指定方法,返回指定的mock内容。

1、项目介绍

imock 包含mock-module、mock-web ,mock-module就是jvm-sandbox的模块,需要安装到目标应用服务器,mock-web 为mock服务后台,imock是前后端分离,还有一个前端react 项目 imcok-web

ume_mock_backend :http://git1.local.umetrip.com/guanhongli/ume_mock_backend

ume_mock_frontend :http://git1.local.umetrip.com/guanhongli/ume_mock_frontend

2、imock使用

首先将前后端项目都跑起来,安装依赖啥的先把环境调通。

2.1、启动imock-web

本地环境:直接idea启动即可

容器环境:java -jar启动

nohup java -jar /opt/applog/MskyLog/mock/mock-web.jar > myout.txt 2>&1 &

2.2、准备mock-module

2.2.0.本地安装

到项目下的bin目录执行 install-local.sh,通过脚本编译mock-module,如需修改代码要重新执行此脚本。

2.2.1.修改cfg

# 心跳上报配置  当环境变量没有配置的时候使用 该配置
# mock 服务的地址和端口
mock.host=http://172.24.146.219:8003
# 标识目标应用的名称
app.name=test-umeapp-checkin
# 标识目标应用的环境
app.env=test-umeapp-checkin

2.2.2.拷贝到sandbox-module目录

将cfg和mock-module拷贝到sandbox/sandbox-module目录下。

cd /opt/applog/MskyLog/mock
cp /opt/applog/MskyLog/mock/mock-module.jar  /opt/app/sandbox/sandbox-module
cp -r /opt/applog/MskyLog/mock/cfg  /opt/app/sandbox/sandbox-module

2.2.3.拷贝到.sandbox-module目录

cp /opt/applog/MskyLog/mock/mock-module.jar /opt/app/.sandbox-module

2.2.4.创建mock日志目录

mkdir -p /home/jboss5/logs/sandbox/mock/

2.3、attch挂载目标应用

jps命令查看pid

cd /opt/app/sandbox/bin/
./sandbox.sh -p 22

2.4、查看log

2.4.1、查看sandbox.log

tailf /opt/applog/MskyLog/sandbox/sandbox.log

2.4.2、查看mock.log

tailf /home/jboss5/logs/sandbox/mock/mock.log

2.4.3、查看服务log

结合log和error日志来看

tailf /opt/applog/MskyLog/UmeCki/UmeCki_info.log
tailf /opt/applog/MskyLog/UmeCki/UmeCki_trace.log
tailf /opt/applog/MskyLog/UmeCki/UmeCki_err.log

3、遇到的问题

3.1、mock切面问题,增加before返回

看代码mock逻辑是在afterReturning中实现的,这样的话是不是原代码逻辑还是会执行,只是修改了返回给调用方的Object或者Exception。比如我想mock调用银行支付逻辑,但是还是会实际支付,所以切面放在before(Advice advice)并且结合returnImmediately会不会效果更好。

作者的代码afterReturning方法是通过advice.getReturnObj().getClass()来获取类,然后把ro.getReturnData()序列化到对象中。

  • 如果before方法,则advice.getReturnObj()为空,空指针无法获取类对象。
  • 如果mock方法的值返回为空,则依然无法获取对象类型。

因此需要换一个方法获取类对象

使用advice.getTarget()方法来获取对象类型,使用advice.getBehavior()获取方法名称。

Method method = advice.getTarget().getClass().getMethod(advice.getBehavior().getName());
returnType = method.getGenericReturnType();
LogUtil.info2("returnType=", String.valueOf(returnType));
Object res1 = JSON.parseObject(ro.getReturnData(), returnType);
LogUtil.info2("res1=", res1.toString());

但经过测试只有springboot可以用,dubbo接口advice.getTarget()为空。

在 Dubbo 中,advice.getTarget() 返回 null 可能是由于 Dubbo 的代理机制导致的。Dubbo 使用代理对象来实现远程服务的调用,代理对象是在运行时动态生成的,而真正的目标对象是通过 Dubbo 的远程调用机制获取的。因此,在 Dubbo 的 Advice 中,advice.getTarget() 返回的是代理对象,而不是真正的目标对象。由于代理对象并不是目标对象本身,因此可能返回 null

在 Spring Boot 中,advice.getTarget() 返回的是目标对象,因为 Spring Boot 使用的代理机制与 Dubbo 不同。Spring Boot 中的 AOP 代理通常是通过 JDK 动态代理或 CGLIB 生成的,这些代理对象会保留对目标对象的引用,因此在 Advice 中调用 advice.getTarget() 可以获取到目标对象的引用,不会返回 null

接着找其他的办法。。。

通过advice.getBehavior().getReturnType()

Method method = advice.getBehavior().getReturnType().getMethod(advice.getBehavior().getName());

完美解决!

// 获取方法的返回对象类型
Object res1 = JSON.parseObject(ro.getReturnData(), advice.getBehavior().getReturnType());
LogUtil.info2("res1=", res1.toString());

3.2、报错time字段不为空

解决办法:修改数据库让字段可以为空。

3.3、imock-web

java.lang.TypeNotPresentException: Type org.springframework.boot.maven.RepackageMojo not present

后续

至此,通过本地调试,二次开发imock已经能够符合我们公司的需求,后续再针对个性化的需求进行开发,更多关于jvm-sandbox imock开发的资料请关注脚本之家其它相关文章!

相关文章

  • SpringCloud微服务基础简介

    SpringCloud微服务基础简介

    今天带大家学习一下SpringCloud微服务的相关知识,文中有非常详细的图文示例及介绍,对正在学习SpringCloud微服务的小伙伴们很有帮助哦,需要的朋友可以参考下
    2021-05-05
  • MyBatis通用的10种写法总结大全

    MyBatis通用的10种写法总结大全

    这篇文章主要给大家介绍了关于MyBatis通用的10种写法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-11-11
  • SpringBoot读取配置的6种方式

    SpringBoot读取配置的6种方式

    本文主要介绍了SpringBoot读取配置的6种方式,主要包括使用默认配置、使用application.properties文件、使用application.yml文件、使用@Value注解、使用Environment对象和使用ConfigurableEnvironment对象,感兴趣的可以了解一下
    2023-08-08
  • java计算指定日期为本年第几周的实例

    java计算指定日期为本年第几周的实例

    本文介绍了在编程中按周统计数据时遇到的问题,由于mysql中按所在月的周数周统计比较麻烦,采用所在年的周数作为分组条件,并通过java计算日期属于年的第一周来进行二次计算,提高性能,同时,作者也分享了计算指定日期为本年第几周的方法,并指出了一些常见方法的缺陷
    2025-11-11
  • java垃圾回收原理之GC算法基础

    java垃圾回收原理之GC算法基础

    本章简要介绍GC的基本原理和相关技术, 下一章节再详细讲解GC算法的具体实现。各种垃圾收集器的实现细节虽然并不相同,但总体而言,垃圾收集器都专注于两件事情:查找所有存活对象,抛弃其他的部分,即死对象,不再使用的对象
    2022-01-01
  • Java使用modbus-master-tcp实现modbus tcp通讯

    Java使用modbus-master-tcp实现modbus tcp通讯

    这篇文章主要为大家详细介绍了另外一种Java语言的modbux tcp通讯方案,那就是modbus-master-tcp,文中的示例代码讲解详细,需要的可以了解下
    2023-12-12
  • idea在用Mybatis时xml文件sql不提示解决办法(提示后背景颜色去除)

    idea在用Mybatis时xml文件sql不提示解决办法(提示后背景颜色去除)

    mybatis的xml文件配置的时候,有时候会没有提示,这让我们很头疼,下面这篇文章主要给大家介绍了关于idea在用Mybatis时xml文件sql不提示的解决办法,提示后背景颜色去除的相关资料,需要的朋友可以参考下
    2023-03-03
  • IDEA设置JVM可分配内存大小和其他参数的教程

    IDEA设置JVM可分配内存大小和其他参数的教程

    这篇文章主要介绍了IDEA设置JVM可分配内存大小和其他参数的教程,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-01-01
  • Java获取当前时间年月日的方法

    Java获取当前时间年月日的方法

    这篇文章主要为大家详细介绍了Java获取当前时间年月日的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-11-11
  • MyBatis实现分页的两种方式

    MyBatis实现分页的两种方式

    本文详细比较了MyBatis中的逻辑分页(如RowBounds)、物理分页(基于数据库的LIMIT/OFFSET)和插件式分页(如PageHelper)的优缺点及应用场景,强调了在处理大数据量时选择物理分页的重要性,以及MyBatis-Plus的分页便利性,对mybatis分页相关知识感兴趣的朋友一起看看吧
    2025-07-07

最新评论