java - Spock-单元测试 :How to write spock unit test for @around annotation which takes Mono

标签 java spring-webflux spring-aop spock

您好,我正在使用以下代码在我的 webflux 应用程序中使用 aop 打印日志,我在编写单元/集成测试时遇到问题?我们可以在这里验证日志交互吗?如有任何帮助,我们将不胜感激

@Retention(RetentionPolicy.RUNTIME) 
@Target(ElementType.METHOD) 
public @interface Loggable {} 
@Aspect
@Slf4j
public class LoggerAspect {

  @Around("@annotation(Loggable)")
  public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable {

    long start = System.currentTimeMillis();
    var result = joinPoint.proceed();
    if (result instanceof Mono) {
      var monoResult = (Mono) result;
      AtomicReference<String> traceId = new AtomicReference<>("");

      return monoResult
        .doOnSuccess(o -> {
          var response = "";
          if (Objects.nonNull(o)) {
            response = o.toString();
          }
          log.info("Enter: {}.{}() with argument[s] = {}",
            joinPoint.getSignature().getDeclaringTypeName(), joinPoint.getSignature().getName(),
            joinPoint.getArgs());
          log.info("Exit: {}.{}() had arguments = {}, with result = {}, Execution time = {} ms",
            joinPoint.getSignature().getDeclaringTypeName(), joinPoint.getSignature().getName(),
            joinPoint.getArgs()[0],
            response, (System.currentTimeMillis() - start));
        });
    }
  }
}

测试失败。 不知怎的,当我调试指针没有进入 doOnNext 方法时。我不确定如何在上面的日志记录方面断言日志交互。在 Junit5 中,我知道我可以对每个方法使用mockito并返回一些东西,但是我如何在 spock 中返回.

class LogAspectTest extends Specification {
  private static final String MOCK_METHOD_LOG_VALUE = "mockMethodLogValue"
  private Logger log = Mock()
  private ProceedingJoinPoint mockJoinPoint = Mock()
  private static Mono<String> methodReturn = Mono.just(["Data", "Data"])
  private LogAspect logAspect = new LogAspect(log)

  @Unroll
  def 'logAround verify log interaction'() {
    given:
    mockJoinPoint.proceed() == Mono.just("Hello")
    final Method method = TestClass.class.getMethod("mockMethod")

    when:
    logAspect.logAround(mockJoinPoint)

    then:
    interaction { mockJoinPointAndMethodSignatureInteractions(method, methodReturnToUse) }

    where:
    resultType | methodReturnToUse
    'Mono'     | methodReturn
  }

  private void mockJoinPointAndMethodSignatureInteractions(Method method, Publisher result) {
    1 * mockJoinPoint.proceed() >> result
    1 * log.info() >> ""

  }

  private static class TestClass {
    @Loggable
    Mono<String> mockMethod() { return Mono.just("data") }

  }
}

是否建议为@Loggable注释编写集成测试,因为它只是记录,不确定如何编写断言日志语句的集成测试

最佳答案

就像我在评论中所说,如果不使用 PowerMock 等附加工具或类似工具,您无法轻松模拟 private static final 字段。我认为每当您需要类似的东西时,您应该重构代码以获得更好的可测试性。这是一个远非完美的想法,但我想给您一个关于如何对方面进行单元测试的想法。至于集成测试,你也可以这样做,但问问自己你想测试什么:真的是切面还是 Spring AOP 切入点匹配工作正常?

无论如何,让我们假设您的测试类是:

package de.scrum_master.stackoverflow.q64164101;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Loggable {}
package de.scrum_master.stackoverflow.q64164101;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import reactor.core.publisher.Mono;

import java.util.Objects;
import java.util.function.Consumer;

@Aspect
public class LogAspect {
  private static final Logger log = LoggerFactory.getLogger(LogAspect.class.getName());

  @Around("@annotation(Loggable)")
  public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable {
    long start = System.currentTimeMillis();
    Object result = joinPoint.proceed();
    if (result instanceof Mono)
      return ((Mono) result).doOnSuccess(getConsumer(joinPoint, start));
    return result;
  }

  public Consumer getConsumer(ProceedingJoinPoint joinPoint, long start) {
    return o -> {
      String response = "";
      if (Objects.nonNull(o))
        response = o.toString();
      log.info("Enter: {}.{}() with argument[s] = {}",
        joinPoint.getSignature().getDeclaringTypeName(), joinPoint.getSignature().getName(),
        joinPoint.getArgs());
      log.info("Exit: {}.{}() had arguments = {}, with result = {}, Execution time = {} ms",
        joinPoint.getSignature().getDeclaringTypeName(), joinPoint.getSignature().getName(),
        joinPoint.getArgs()[0],
        response, (System.currentTimeMillis() - start));
    };
  }
}

看看我如何将 lambda 分解为辅助方法?它有两个作用:

  • 它使 logAround(ProceedingJoinPoint) 建议方法更具可读性。
  • 它允许您对辅助方法进行 stub ,而无需验证日志记录是否已完成,您只需验证是否为 Mono 结果调用了辅助方法(而不是为其他结果类型调用)。<

最简单形式的测试可能如下所示:

package de.scrum_master.stackoverflow.q64164101

import org.aspectj.lang.ProceedingJoinPoint
import reactor.core.publisher.Mono
import spock.lang.Specification

class LogAspectTest extends Specification {
  LogAspect logAspect = Spy()
  ProceedingJoinPoint joinPoint = Mock()

  def "aspect target method returns a Mono"() {
    given:
    joinPoint.proceed() >> Mono.just("Hello")

    when:
    logAspect.logAround(joinPoint)

    then:
    1 * logAspect.getConsumer(joinPoint, _)
  }

  def "aspect target method does not return a Mono"() {
    given:
    joinPoint.proceed() >> "dummy"

    when:
    logAspect.logAround(joinPoint)

    then:
    0 * logAspect.getConsumer(joinPoint, _)
  }
}

请注意我如何使用 Spy (即基于原始对象的部分模拟)来有选择地 stub 辅助方法。


更新:更集成测试的另一种选择是配置您的日志框架以登录到您可以控制和验证的目标,例如登录到内存数据库或您可以访问的缓冲区。

关于java - Spock-单元测试 :How to write spock unit test for @around annotation which takes Mono,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64164101/

相关文章:

java - @Transactional 是否打开 session 并绑定(bind)它?

java - 在@Around Aspect 中重新抛出异常

Java 6/7反编译命令行

java - 如何从 ServerResponse 获取字符串形式的正文进行测试?

spring - Spring WebFlux 中的 ReactiveSecurityContextHolder 为空

project-reactor - Spring Reactor - 等到一个单声道完成然后做下一个单声道

java - 如何为ProxyFactoryBean设置多个目标?

java - Android:将 VectorDrawable 添加到 ImageView

java - Hibernate 教程不起作用

java - 如何让应用程序中的所有组件响应特定的按键事件?