3

我正在使用 JMockit 测试应用程序(Java 8、Junit 4.12、JMockit 1.17)。

我有一些代码应该将一些数据上传到端点列表。

实际执行上传到端点的类称为 PlatformDataUploader。此类有一个方法,称为“上传”,它执行单个“上传”到单个端点。此方法采用目的地的名称(这是一个两个字符串,以及要上传的对象)。根据目的地的名称,它构造数据应该发送到的 URL。

我想创建一个测试来简单地确认,当应用程序执行时,PlatformDataUploader 的上传方法被调用了预期的次数(7 次,因为应用程序当前配置为将数据上传到 7 个端点)。我想确认传递给上传方法的目标字符串是预期的,但我不关心发送的数据(由 PlatformInstallationData 的实例表示)。

应用程序代码的简化版本如下:

 ...

 private boolean uploadToServices(final List<String> serviceNames) {

    boolean allGood = true;

    PlatformDataUploader platformDataUploader = new PlatformDataUploader();

    for (String serviceName : serviceNames) {

        LOG.info("Attempting to upload to " + serviceName + "...");

        // construct object to send
        PlatformInstallationData platformInstallationData = new PlatformInstallationData();

        ...
        // code here that adds content to platformInstallationData
        ...

        // send object to endpoint of this service
        allGood = allGood &&
            platformDataUploader.upload(serviceName, platformInstallationData);
    }

    return allGood;
}

测试代码的简化版本如下:

@Test
public void whenUploadThenExpectedCallsToUploader(@Mocked final PlatformDataUploader platformDataUploader,
        @Mocked final PlatformInstallationData platformInstallationData)
                throws IOException {

    UploaderApplication target = new UploaderApplication();

    new Expectations() {
        {
            platformDataUploader.upload("AP", platformInstallationData);
            result = true;
            times = 1;

            platformDataUploader.upload("VV", platformInstallationData);
            result = true;
            times = 1;
            ...
            THE REST OF THE EXPECTED CALLS 
            ...


        }
    };

    target.execute(params);
}

当我执行测试时,我得到:“mockit.internal.MissingInvocation: Missing 1 invocation”。它指的是我所定义的期望中的第一行。

问题似乎是上传方法中的第二个参数,它是一个模拟的 PlatformInstallationData,与应用程序代码中创建的实例不匹配(即使该实例也应该被模拟)。

为了试图了解发生了什么,我做了一个测试,我从上传方法(从应用程序代码和测试代码)中删除了第二个参数,因此它只需要匹配一个字符串,例如“AP”反对那个时候应用程序中存在的字符串,在这种情况下它匹配正确。所以我知道,使用这种方法,它可以正确匹配一个字符串,但它无法匹配模拟的 PlatformInstallationData 对象。

我对这种行为感到惊讶,因为我确信我以前能够创建包含模拟对象作为方法参数的期望,并且我认为它正确地匹配了它们。

谁能解释一下这种行为,并可能建议如何测试。非常感谢!

4

1 回答 1

1

我找到了一种编写测试的方法,使其与调用匹配,并确认对上传方法进行了预期的调用次数,并且预期的字符串作为第一个参数传递。我对第二个参数使用了“任何”占位符。这有点难看,因为我必须做演员。如下所示:

@Test
public void whenUploadThenExpectedCallsToUploader(@Mocked final PlatformDataUploader platformDataUploader,
        @Mocked final PlatformInstallationData platformInstallationData)
                throws IOException {

    InstallPlatformCommand target = new InstallPlatformCommand(Action.INSTALL_PLATFORM);

    new Expectations() {
        {
            platformDataUploader.upload("AP", (PlatformInstallationData) any);
            result = true;
            times = 1;

            platformDataUploader.upload("VV", (PlatformInstallationData) any);
            result = true;
            times = 1;

            ...
            The rest of the expected calls
            ...
        }
    };

    target.execute(params);
}

我相信还有其他方法可以做到这一点。如果您有更清洁的方法,请告诉我。

于 2015-12-10T09:24:43.943 回答