Advanced JUnit

advertisement
JUnit
Automated Software Testing Framework
Advanced Material
Paul Ammann & Jeff Offutt
http://www.cs.gmu.edu/~offutt/softwaretest/
Advanced Topics in JUnit
• Assertion Patterns
– How To Decide If Your Test Passes
– State Testing vs. Interaction Testing Patterns
• Parameterized JUnit Tests
– How to Describe and Run Very Similar Tests
• JUnit Theories
– Applying the Contract Model to Testing
– AAA Model: Assume, Act, Assert
– Very Powerful Approach
• But Also Still A Work in Progress
Introduction to Software Testing (Ch 1)
© Ammann & Offutt
2
Assertion Patterns for Unit Tests
• State Testing Patterns
– Final State Assertion
• Most Common Pattern: Arrange. Act. Assert.
– Guard Assertion
• Assert Both Before and After The Action (Precondition Testing)
– Delta Assertion
• Verify a Relative Change to the State
– Custom Assertion
• Encodes Complex Verification Rules
• Interaction Assertions
– Verify Expected Interactions
– Heavily used in Mocking tools
– Very Different Analysis Compared to State Testing
• Resource: http://martinfowler.com/articles/mocksArentStubs.html
Introduction to Software Testing (Ch 1)
© Ammann & Offutt
3
Parameterized JUnit Tests
• Problem: Testing A Function With Similar Values
– How To Avoid Test Code Bloat?
• Simple Example: Adding Two Numbers
– Adding a Given Pair of Numbers Is Just Like Adding Any
Other Pair
– You Really Only Want to Write One Test
• Parameterized Unit Tests Call Consructor For Each
Logical Set of Data Values
– Same Tests Are Then Run On Each Set of Data Values
– List of Data Values Identified with @Parameters
Annotation
Introduction to Software Testing (Ch 1)
© Ammann & Offutt
4
Parameterized Unit Tests
import org.junit.*;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameters;
import static org.junit.Assert.*;
import java.util.*;
@RunWith(Parameterized.class)
public class ParamTest {
public int sum, a, b;
public ParamTest (int sum, int a, int b) { this.sum = sum; this.a = a; this.b = b; }
@Parameters public static Collection<Object[]> parameters() {
return Arrays.asList (new Object [][] {{0, 0, 0}, {2, 1, 1}});
}
@Test public void additionTest() { assertEquals(sum, a+b); }
}
Introduction to Software Testing (Ch 1)
© Ammann & Offutt
5
JUnit Theories
• These Are Unit Tests With Actual Parameters
– So Far, We’ve Only Seen Parameterless Test Methods
• Contract Model: Assume, Act, Assert
– Assumptions (Preconditions) Limit Values Appropriately
– Action Performs Activity Under Scrutiny
– Assertions (Postconditions) Check Result
@Theory public void removeThenAddDoesNotChangeSet(
Set<String> set, String string) {
// Parameters!
assumeTrue(set.contains(string)) ;
// Assume
Set<String> copy = new HashSet<String>(set); // Act
copy.remove(string);
copy.add(string);
assertTrue (set.equals(copy));
// Assert //
// System.out.println(“Instantiated test: “ + set + “, “ + string);
}
Introduction to Software Testing (Ch 1)
© Ammann & Offutt
6
Question: Where Does The Data Come
From?
• Answer:
– All Combinations of Values from @DataPoint Annotations Where
Assume Clause is True
– Four (of Nine) Combinations in This Particular Case
– Note: @DataPoint Format is an Array.
@DataPoints
public static String[] string = {"ant", "bat", "cat"};
@DataPoints
public static Set[] sets = {
new HashSet(Arrays.asList("ant", "bat")),
new HashSet(Arrays.asList(“bat", “cat", “dog“, “elk”)),
new HashSet(Arrays.asList(“Snap”, “Crackle”, “Pop"))
};
Introduction to Software Testing (Ch 1)
© Ammann & Offutt
7
JUnit Theories Need BoilerPlate
import org.junit.*;
import org.junit.runner.RunWith;
import static org.junit.Assert.*;
import static org.junit.Assume.*;
import org.junit.experimental.theories.DataPoint;
import org.junit.experimental.theories.DataPoints;
import org.junit.experimental.theories.Theories;
import org.junit.experimental.theories.Theory;
import java.util.*;
@RunWith(Theories.class)
public class SetTheoryTest {
… // See Earlier Slides
}
Introduction to Software Testing (Ch 1)
© Ammann & Offutt
8
Download