Best way of logging exceptions when tests fail (e.g. using a junit rule)

前端 未结 3 1670
孤独总比滥情好
孤独总比滥情好 2021-01-13 07:54

When I\'m running a complete test suite, it would be helpful if exceptions that caused a test to fail would appear in my (SLF4J-)log. What is the best method to achieve this

相关标签:
3条回答
  • 2021-01-13 08:25

    Look outside the box... What are you using to run the tests? Most environments for running tests (e.g., Ant, Jenkins, Maven, etc.) have means for using a testrunner that outputs an XML file, with support for aggregating the XML files from a suite into a comprehensive report.

    0 讨论(0)
  • 2021-01-13 08:30

    This seems so easy that I think I am wrong and you are asking something different, but maybe I can help:

    JUnit 4.X

    @Test(expected=Exception.class)

    is going to pass the test if the exception is thrown in the test, or fail with a message captured by the Junit framework

    0 讨论(0)
  • 2021-01-13 08:36

    You need to extend TestRule, in particular the apply(). For an example, have a look at org.junit.rules.ExternalResource & org.junit.rules.TemporaryFolder.

    ExternalResource looks like this:

    public abstract class ExternalResource implements TestRule {
        public Statement apply(Statement base, Description description) {
            return statement(base);
        }
    
        private Statement statement(final Statement base) {
            return new Statement() {
                @Override
                public void evaluate() throws Throwable {
                    before();
                    try {
                        base.evaluate();
                    } finally {
                        after();
                    }
                }
            };
        }
    
        /**
         * Override to set up your specific external resource.
         * @throws if setup fails (which will disable {@code after}
         */
        protected void before() throws Throwable {
            // do nothing
        }
    
        /**
         * Override to tear down your specific external resource.
         */
        protected void after() {
            // do nothing
        }
    }
    

    TemporaryFolder then extends this and implements before() and after().

    public class TemporaryFolder extends ExternalResource {
        private File folder;
    
        @Override
        protected void before() throws Throwable {
            // create the folder
        }
    
        @Override
        protected void after() {
            // delete the folder
        }
    

    So the before gets called before the testMethod, and the after is called in the finally, but you can catch and log any Exception, like:

        private Statement statement(final Statement base) {
            return new Statement() {
                @Override
                public void evaluate() throws Throwable {
                    before();
                    try {
                        base.evaluate();
                    } catch (Exception e) {
                        log.error("caught Exception", e);
                    } finally {
                        after();
                    }
                }
            };
        }
    

    EDIT: The following works:

    public class SoTest {
        public class ExceptionLoggingRule implements TestRule {
            public Statement apply(Statement base, Description description) {
                return statement(base);
            }
    
            private Statement statement(final Statement base) {
                return new Statement() {
                    @Override
                    public void evaluate() throws Throwable {
                        try {
                            base.evaluate();
                        } catch (Exception e) {
                            System.out.println("caught an exception");
                            e.printStackTrace(System.out);
                            throw e;
                        }
                    }
                };
            }
        }
    
        @Rule public ExceptionLoggingRule exceptionLoggingRule = new ExceptionLoggingRule();
        @Rule public ExpectedException expectedException = ExpectedException.none();
    
        @Test
        public void testMe() throws Exception {
            expectedException.expect(IOException.class);
            throw new IOException("here we are");
        }
    }
    

    The test passes and you get the following output:

    caught an exception
    java.io.IOException: here we are
        at uk.co.farwell.junit.SoTest.testMe(SoTest.java:40)
        at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
        at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
    ...
    

    The order that the rules are applied is ExpectedException which calls ExceptionLoggingRule which calls the testMe method. The ExceptionLoggingRule catches the Exception, logs it and rethrows it, and it is then processed by ExpectedException.

    If you want to log only unexpected exceptions, you just switch the declaration order of the rules:

        @Rule public ExpectedException expectedException = ExpectedException.none();
        @Rule public ExceptionLoggingRule exceptionLoggingRule = new ExceptionLoggingRule();
    

    That way, expectedException is applied first (i.e. nested in exceptionLoggingRule), and only rethrows exceptions that are not expected. Furthermore, if some exception was expected and none occured, expectedException will throw an AssertionError which will also get logged.

    This evaluation order isn't guaranteed, but it is quite unlikely to vary unless you're playing with very different JVMs, or inheriting between Test classes.

    If the evaluation order is important, then you can always pass one rule to the other for evaluation.

    EDIT: With the recently released Junit 4.10, you can use @RuleChain to chain rules correctly:

    public static class UseRuleChain {
       @Rule
       public TestRule chain= RuleChain
                              .outerRule(new LoggingRule("outer rule")
                              .around(new LoggingRule("middle rule")
                              .around(new LoggingRule("inner rule");
    
       @Test
       public void example() {
               assertTrue(true);
       }
    }
    

    writes the log

    starting outer rule
    starting middle rule
    starting inner rule
    finished inner rule
    finished middle rule
    finished outer rule
    
    0 讨论(0)
提交回复
热议问题