Stubing when initializing fails when stub returns 存根

Stubbing when initializing fails when stub returns stub

我有以下 Spring 代码来测试 Spock:

@Service
@RequiredArgsConstructor
public class MyService {
  private final RestTemplateBuilder restTemplateBuilder;
  // ...

  public Path downloadFile(String url) {
    try {
      ResponseEntity<byte[]> response = buildRestTemplate().getForEntity(url, byte[].class);
      File tempFileZip = File.createTempFile("myTempFile", ".zip");
      FileUtils.writeByteArrayToFile(tempFileZip, response.getBody());
      return tempFileZip.toPath();
    } catch (Exception e) {
      // ...
    }
  }

  private RestTemplate buildRestTemplate() {
    SimpleClientHttpRequestFactory requestFactory = new SimpleClientHttpRequestFactory();
    Proxy proxy = new Proxy(/*...*/);
    requestFactory.setProxy(proxy);
    return restTemplateBuilder.requestFactory(() -> requestFactory).build();
  }
}

在此代码中,下载了一个文件。我的 test/resources 文件夹中有一个示例文件,我想将其提供给存根下载过程的结果。这是我可以让它工作的一种方式,我对此有一些疑问:

package de.scrum_master.Whosebug.q59630155

class MySpec extends Specification {

  RestTemplate restTemplate = Stub {
    getForEntity(_, byte[].class) >>
      ResponseEntity.ok(
        Files.readAllBytes(
          Paths.get(
            getClass()
              .getResource("/data/MySample/MySample.zip")
              .toURI()
          )
        )
      )
  }
  RestTemplateBuilder restTemplateBuilder = Stub()
  def myService = new MyService(restTemplateBuilder)

  def setup() {
    with(restTemplateBuilder) {
      requestFactory(_) >> restTemplateBuilder
      build() >> restTemplate
    }
  }

  def "downloadFile"() {
    when:
    def response = myService.downloadFile()

    then:
    // ...

    cleanup:
    Files.delete(response)
  }
}

A) 我想更好地理解为什么资源文件的路径通过调试器评估为 ...target/test-classes/data/ebaRegistrySample/ebaRegistrySample.zip 而不是包含 /test/resources 的东西是因为 maven ?

编辑:评论中已经回答这是因为 maven inded。

B) 在 setup() 或测试中存根 RestTemplateBuilder 工作正常。也可以在其中存根 RestTemplate 或在规范中初始化时如图所示。但是当我还在规范中的初始化时存根 RestTemplateBuilder 时:

package de.scrum_master.Whosebug.q59630155

class MySpec extends Specification {
  // ...

  RestTemplateBuilder restTemplateBuilder = Stub {
      requestFactory(_) >> restTemplateBuilder
      build() >> restTemplate
  }

  // ...
}

我得到一个

java.lang.NullPointerException: null

当我这样做时我得到了同样的异常

build() >> null

我怀疑在 Spec 初始化时对 restTemplate 和 restTemplateBuilder 进行存根会导致 restTamplate 在 restTemplateBuilder 尝试访问它时还没有值。我认为在初始化时像这样链接存根似乎应该有效,所以它可能是一个错误。

这是一个错误还是有其他原因导致它不起作用?

编辑:当我在初始化后单独存根 requestFactory 方法时它确实有效。所以这只是 C 案例的一个不同错误。

C) 在初始化 RestTemplateBuilder 时存根通常不起作用。虽然前一个案例导致异常,但将其放在 setup() 或测试中:

package de.scrum_master.Whosebug.q59630155

class MySpec extends Specification {

  def "downloadFile"() {
    setup:
    RestTemplate restTemplate = Stub {
    getForEntity(_, byte[].class) >>
      ResponseEntity.ok(
        Files.readAllBytes(
          Paths.get(
            getClass()
              .getResource("/data/MySample/MySample.zip")
              .toURI()
          )
        )
      )
    }
    RestTemplateBuilder restTemplateBuilder = Stub() {
      requestFactory(_) >> restTemplateBuilder
      build() >> restTemplate
    }
    def myService = new MyService(restTemplateBuilder)

    // ...
  }
}

实际导致编译错误:

requestFactory(_) >> restTemplateBuilder
No candidates found for method call restTemplateBuilder

编辑:一位同事建议这实际上是有意为之,因为它可以防止堆栈中的无限递归内存分配。

这是一个错误还是它不起作用的其他原因?

再看看自己的代码:

RestTemplateBuilder restTemplateBuilder = Stub {
  requestFactory(_) >> restTemplateBuilder
  build() >> restTemplate
}

您试图在 restTemplateBuilder 存根对象创建之前存根一个返回 restTemplateBuilder 的方法,它是一个自引用,类似于写 String text = "xy" + text。这也解释了为什么它分两步工作:首先创建存根实例,然后存根引用先前创建的对象的方法。

没有错误。问题出在键盘前面。没有冒犯的意思,这也发生在我身上。 ;-)