Determining which test cases covered a method

前端 未结 2 691
北海茫月
北海茫月 2021-01-14 23:34

The current project I\'m working on requires me to write a tool which runs functional tests on a web application, and outputs method coverage data, recording which test case

2条回答
  •  不知归路
    2021-01-15 00:10

    Your proposed solution seems like a reasonable one, except for the proposed solution to relate the test and request by timing. I've tried to do this sort of thing before, and it works. Most of the time. Unless you write your JUnit code very carefully, you'll have lots of issues, because of differences in time between the two machines, or if you've only got one machine, just matching one time against another.

    A better solution would be to implement a Tomcat Valve which you can insert into the lifecycle in the server.xml for your webapp. Valves have the advantage that you define them in the server.xml, so you're not touching the webapp at all.

    You will need to implement invoke(). The best place to start is probably with AccessLogValve. This is the implementation in AccessLogValve:

    /**
     * Log a message summarizing the specified request and response, according
     * to the format specified by the pattern property.
     *
     * @param request Request being processed
     * @param response Response being processed
     *
     * @exception IOException if an input/output error has occurred
     * @exception ServletException if a servlet error has occurred
     */
    public void invoke(Request request, Response response) throws IOException,
            ServletException {
    
        if (started && getEnabled()) {                
            // Pass this request on to the next valve in our pipeline
            long t1 = System.currentTimeMillis();
    
            getNext().invoke(request, response);
    
            long t2 = System.currentTimeMillis();
            long time = t2 - t1;
    
            if (logElements == null || condition != null
                    && null != request.getRequest().getAttribute(condition)) {
                return;
            }
    
            Date date = getDate();
            StringBuffer result = new StringBuffer(128);
    
            for (int i = 0; i < logElements.length; i++) {
                logElements[i].addElement(result, date, request, response, time);
            }
    
            log(result.toString());
        } else
            getNext().invoke(request, response);       
    }
    

    All this does is log the fact that you've accessed it.

    You would implement a new Valve. For your requests you pass a unique id as a parameter for the URL, which is used to identify the tests that you're running. Your valve would do all of the heavy lifting before and after the invoke(). You could remove remove the unique parameter for the getNext().invoke() if needed.

    To measure the coverage, you could use a coverage tool as suggested by JB Nizet, based on the unique id that you're passing over.

    So, from junit, if your original call was

    @Test void testSomething() {
        selenium.open("http://localhost/foo.jsp?bar=14");
    }
    

    You would change this to be:

    @Test void testSomething() {
        selenium.open("http://localhost/foo.jsp?bar=14&testId=testSomething");
    }
    

    Then you'd pick up the parameter testId in your valve.

提交回复
热议问题