unit-testing - 此时缺少对模拟类型的调用;

标签 unit-testing jmockit injectable

我是 jMockit 新手。我试图在方法中模拟 java.io.File 类型的多个实例。在某些地方,我不应该模拟文件对象。因此,我使用 @Injectable。它抛出以下异常。

我不想模拟 java.io.File 的所有实例。我希望从方法返回的实例是实际的文件。

下面是测试类。

/**
 * 
 */
package org.iis.uafdataloader.tasklet;

import static org.junit.Assert.fail;

import java.io.File;
import java.io.FilenameFilter;
import java.io.IOException;
import java.util.regex.Pattern;

import mockit.Expectations;
import mockit.Injectable;
import mockit.Mocked;
import mockit.NonStrictExpectations;
import mockit.VerificationsInOrder;

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.filefilter.RegexFileFilter;
import org.iis.uafdataloader.tasklet.validation.FileNotFoundException;
import org.junit.Test;
import org.springframework.batch.core.StepContribution;
import org.springframework.batch.core.scope.context.ChunkContext;
import org.springframework.batch.repeat.RepeatStatus;

/**
 * @author K23883
 *
 */
public class FileMovingTaskletTest {

    private FileMovingTasklet fileMovingTasklet;

    @Mocked
    private StepContribution contribution;

    @Mocked
    private ChunkContext chunkContext;



    /**
     * Test method for
     * {@link org.iis.uafdataloader.tasklet.FileMovingTasklet#execute(org.springframework.batch.core.StepContribution, org.springframework.batch.core.scope.context.ChunkContext)}
     * .
     * 
     * @throws Exception
     */
    @Test
    public void testExecuteWhenWorkingDirDoesNotExist(
//          @Mocked final File file,
            @Injectable final File sourceDirectory,
            @Injectable final File workingDirectory,
            @Injectable final File archiveDirectory,
            @Mocked final RegexFileFilter regexFileFilter,
            @Mocked final FileUtils fileUtils) throws Exception {

        fileMovingTasklet = new FileMovingTasklet();
        fileMovingTasklet.setSourceDirectoryPath("sourceDirectoryPath");
        fileMovingTasklet.setInFileRegexPattern("inFileRegexPattern");
        fileMovingTasklet.setArchiveDirectoryPath("archiveDirectoryPath");
        fileMovingTasklet.setWorkingDirectoryPath("workingDirectoryPath");



        final File[] sourceDirectoryFiles = new File[] {
                new File("sourceDirectoryPath/ISGUAFFILE.D140728.C00"),
                new File("sourceDirectoryPath/ISGUAFFILE.D140729.C00") };

        final File[] workingDirectoryFiles = new File[] {
                new File("workingDirectoryPath/ISGUAFFILE.D140728.C00"),
                new File("workingDirectoryPath/ISGUAFFILE.D140729.C00") };




        new NonStrictExpectations(){{

                new File("sourceDirectoryPath");
                result = sourceDirectory;

                sourceDirectory.exists();
                result = true;

                sourceDirectory.isDirectory();
                result = true;

                // workingDirectory =
                new File("workingDirectoryPath");
                result = workingDirectory;

                workingDirectory.exists();
                result = false;

                workingDirectory.mkdirs();

                FileUtils.cleanDirectory(onInstance(workingDirectory));

                FilenameFilter fileNameFilter = new RegexFileFilter(anyString,
                        Pattern.CASE_INSENSITIVE);

                sourceDirectory.listFiles(fileNameFilter);
                result = sourceDirectoryFiles;

                System.out.println("sourceDirectoryFile :"
                        + ((File[]) sourceDirectoryFiles).length);

                // for (int i = 0; i < sourceDirectoryFiles.length; i++) {
                // FileUtils.moveFileToDirectory(sourceDirectoryFiles[i],
                // workingDirectory, true);
                // }

                // archiveDirectory =
                new File("archiveDirectoryPath");
                result = archiveDirectory;

                workingDirectory.listFiles();
                result = workingDirectoryFiles;

                // for (int i = 0; i < workingDirectoryFiles.length; i++) {
                // FileUtils.copyFileToDirectory(workingDirectoryFiles[i],
                // archiveDirectory);
                // }

            }};


            RepeatStatus status = fileMovingTasklet.execute(contribution,
                    chunkContext);
            assert (status == RepeatStatus.FINISHED);


        new VerificationsInOrder() {{

            sourceDirectory.exists();
            onInstance(sourceDirectory).isDirectory();
            onInstance(workingDirectory).exists();

            onInstance(workingDirectory).mkdirs();

            onInstance(sourceDirectory).listFiles((FilenameFilter)any);

            FileUtils.moveFileToDirectory((File)any, onInstance(workingDirectory), true); 
            times = 2;

            FileUtils.copyFileToDirectory((File)any, onInstance(archiveDirectory));
            times= 2;

        }};

    }

}

下面是实际的实现方法

/*
 * (non-Javadoc)
 * 
 * @see org.springframework.batch.core.step.tasklet.Tasklet#execute(org.
 * springframework.batch.core.StepContribution,
 * org.springframework.batch.core.scope.context.ChunkContext)
 */
@Override
public RepeatStatus execute(StepContribution contribution,
        ChunkContext chunkContext) throws Exception {

    File sourceDirectory = new File(sourceDirectoryPath);

    if (sourceDirectory == null || !sourceDirectory.exists()
            || !sourceDirectory.isDirectory()) {
        throw new FileNotFoundException("The source directory '"
                + sourceDirectoryPath
                + "' doesn't exist or can't be read or not a directory");
    }


    File workingDirectory = new File(workingDirectoryPath);

    if (workingDirectory != null && !workingDirectory.exists() ) {
        workingDirectory.mkdirs();

    }

    FileUtils.cleanDirectory(workingDirectory);

    FilenameFilter fileFilter = new RegexFileFilter(inFileRegexPattern,
            Pattern.CASE_INSENSITIVE);

    File[] sourceDirectoryFiles = sourceDirectory.listFiles(fileFilter);

    System.out.println("sourceDirectoryFiles : " + sourceDirectoryFiles.length);

    for (File file : sourceDirectoryFiles) {
        FileUtils.moveFileToDirectory(file, workingDirectory, true);
    }

    File archiveDirectory =  new File(archiveDirectoryPath);

    for (File file : workingDirectory.listFiles()) {
        FileUtils.copyFileToDirectory(file, archiveDirectory);
    }

    return RepeatStatus.FINISHED;
}

下面是堆栈跟踪。

java.lang.IllegalStateException: Missing invocation to mocked type at this point; please make sure such invocations appear only after the declaration of a suitable mock field or parameter
    at org.iis.uafdataloader.tasklet.FileMovingTaskletTest$1.<init>(FileMovingTaskletTest.java:75)
    at org.iis.uafdataloader.tasklet.FileMovingTaskletTest.testExecuteWhenWorkingDirDoesNotExist(FileMovingTaskletTest.java:71)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:483)
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)

请帮我解决这个问题。

最佳答案

@Injectable 为您提供一个模拟实例;它不会影响模拟类型的其他实例。因此,当测试尝试记录 new File("sourceDirectoryPath") 时,它会显示“此时缺少对模拟类型的调用”,因为 File(String)没有被 mock 。

要模拟整个 File 类(包括其构造函数)以使所有实例都受到影响,您需要使用 @Mocked 来代替,如以下示例所示:

@Test
public void mockFutureFileObjects(@Mocked File anyFile) throws Exception
{
    final String srcDirPath = "sourceDir";
    final String wrkDirPath = "workingDir";

    new NonStrictExpectations() {{
        File srcDir = new File(srcDirPath);
        srcDir.exists(); result = true;
        srcDir.isDirectory(); result = true;

        File wrkDir = new File(wrkDirPath);
        wrkDir.exists(); result = true;
    }};

    sut.execute(srcDirPath, wrkDirPath);
}

JMockit 教程 describes the same mechanism ,尽管语法略有不同。

也就是说,我建议使用真实的文件和目录编写测试。

关于unit-testing - 此时缺少对模拟类型的调用;,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25049264/

相关文章:

java - 使用 Mockito 的 argThat() 匹配器导致 "method ambiguous for the type"错误

javascript - 单元测试 - 通量和数据持久性

maven - Powermock 和 JMockit 单元测试的测试覆盖率

ViewHelper 新/可注入(inject)的困境

flutter - 将生成的模拟类注入(inject)测试“环境”

c# - 无法设置 TestContext 属性

python - 我什么时候应该使用 setUpClass 什么时候使用 __init__?

java - jmockit类强制转换异常

java - 使用jmockit测试main方法

angular - 使用 jasmine angular2 注入(inject)私有(private)服务的单元测试