【问题标题】:Reduce JUnit Output减少 JUnit 输出
【发布时间】:2026-02-10 07:25:02
【问题描述】:

我正在使用以下方法以编程方式运行 JUnit 测试:

String res = runProcess("java -classpath " + dirPath + ";" + classpath + " org.junit.runner.JUnitCore "+ getClassName(filePath));

private static String runProcess(String command) throws Exception {
    Process process = Runtime.getRuntime().exec(command);
    String inputStr = IOUtils.toString(process.getInputStream());
    process.waitFor();
    return inputStr;
}

但是,它会输出很多附加信息,例如失败次数、运行所用时间和 junit 信息。

示例输出如下所示:

JUnit version 4.12
.E
Time: 0.011
There was 1 failure:
1) test10(ErrorTestLang)
java.lang.AssertionError: Contract failed: compareTo-equals on fraction1 and fraction4
    at org.junit.Assert.fail(Assert.java:88)
    at org.junit.Assert.assertTrue(Assert.java:41)
    at ErrorTestLang.test10(ErrorTestLang.java:32)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
    at java.lang.reflect.Method.invoke(Unknown Source)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.junit.runners.Suite.runChild(Suite.java:128)
    at org.junit.runners.Suite.runChild(Suite.java:27)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:115)
    at org.junit.runner.JUnitCore.runMain(JUnitCore.java:77)
    at org.junit.runner.JUnitCore.main(JUnitCore.java:36)

FAILURES!!!
Tests run: 1,  Failures: 1

我的目标是以编程方式多次运行 JUnit 测试套件,因为我以编程方式修改它并确保结果仍然相同。也就是说,即使我删除了一些未使用/“死”的代码行,所有相同的断言都会失败/通过。

因此,我对输出的部分感兴趣:

java.lang.AssertionError: Contract failed: compareTo-equals on fraction1 and fraction4

因为我会将其与其他结果进行比较,并且不想考虑运行时间或行号等细节。

有没有让JUnit输出更简洁的标志?

【问题讨论】:

  • 我不明白您为什么要以这种方式启动 JUnit。几乎所有 IDE 都有一个运行 JUnit Test 的插件,您可以像自己的“启动器”一样轻松地启动它。还有一些插件会在每次更改代码库后运行受影响的测试(例如,infinitest 用于 eclipse 和 netbeans)...

标签: java unit-testing parsing testing junit


【解决方案1】:

我不认为您可以通过这种方式使 JUnit 更简洁,毕竟您正在运行一个外部 Java 程序,它会产生异常并且该异常会出现在屏幕上。从技术上讲,它甚至不是例外,而是java.lang.Error

org.junit.Assert#fail 是这样实现的:

public static void fail(String message) {
    if(message == null) {
        throw new AssertionError();
    } else {
        throw new AssertionError(message);
    }
}

当然你会解析流的输出,但让我建议你一种不同的方法:

您可以通过包装器运行它,而不是直接运行JUnitCore,并且在此包装器中,您可以添加一个侦听器,该侦听器将跟踪测试执行并能够对失败/成功执行等事件做出反应。

Junit 中有一个RunListener 的概念可以处理这个问题:

 public class MyWrapperToRun {
   public void main(String... args) {
      JUnitCore core= new JUnitCore();
      core.addListener(new MyWrappingListener());
      core.run(MyTestClass.class);   // I don't have access to MyTestClass.class
   }
 } 

 public class MyWrappingListener implements RunListener {
   // react on events emitted by junit in methods of this listener
 }

在侦听器实现中,您可以将结果存储在文件甚至数据库中以供将来比较。

【讨论】:

  • 感谢您的建议!这里的问题是我无权访问对象的类。这是我程序的一个完全外部的 Java 文件,我可以获取类名,但不能获取表示程序中 Java 对象类型的 Class 对象。
  • 不确定我是否明白你的意思。 'runProcess' 方法是你的代码,你知道你运行 JUnitCore 的'main' 方法吗?假设您现在已经构建了类路径 X。我建议使用包装器和侦听器构造类路径 Y = X + 模块。包装器将有一个 main 方法,它将像以前一样运行相同的 JUnit 核心,但不是通过 main 方法。可能您可以像以前一样将类作为字符串参数运行。因此包装器将成为外部进程的一部分,但该进程不会意识到这一事实。
【解决方案2】:

目前,如上所述运行 JUnit 测试输出会产生一致的输出。

我们可以获取有关失败的方法以及失败的断言的信息。

这由 x) 表示,其中 x 是第 x 个失败的断言。

/**
 * Minimize the String output obtained from running a JUnit test suite
 * 
 * @param input
 *            The String produced from running a JUnit test suite
 * @return A String that has been minimized to contain the method that
 *         contains the failing assertion and the assertion that failed
 */
private static String minimizeJUnitOutput(String input) {
    Scanner scn = new Scanner(input);

    // JUnit output starts with index 1 for first failure
    int index = 1;
    // String to represent the result that we return
    String res = "";
    // Continue until we finish processing the input String
    while (scn.hasNextLine()) {
        String line = scn.nextLine();
        // Look for a String of the form i) where i is the current failing
        // test index
        if (line.startsWith(index + ") ")) {
            res += line + "\n" + scn.nextLine() + "\n";
            index++;
        }
    }
    scn.close();
    return res;
}

前面比较好的解决了这个问题。

【讨论】: