001/** 002 * This is part of HW0: Environment Setup and Java Introduction for CSE 331. 003 */ 004package hw0.test; 005import hw0.*; 006 007import java.util.Iterator; 008import java.util.LinkedList; 009import java.util.Random; 010import java.util.Set; 011import java.util.HashSet; 012 013import org.junit.Test; 014import org.junit.BeforeClass; 015import static org.junit.Assert.*; 016import static org.hamcrest.Matchers.greaterThan; 017 018/** 019 * BoxTest is a glassbox test of the Box class. 020 * 021 * Recall that like a BallContainer, the Box is a container for Balls and you 022 * can only put a Ball into a Box once. After you put the Ball into 023 * the Box, further attempts to do so will fail, since the Ball is 024 * already in the Box! Similarly, you cannot expect to remove a Ball 025 * from a Box if it is not inside the Box. 026 * 027 * In addition, a Box differs from a ballcontainer because it only has a finite 028 * volume. As Balls get put into a Box, it gets filled up. Once a Box 029 * is full, further attempts to put Balls into the Box will also fail. 030 * 031 * @see hw0.Ball 032 * @see hw0.BallContainer 033 * @see hw0.Box 034 */ 035public class BoxTest { 036 037 private static Box box = null; 038 private static Ball[] b = null; 039 040 private static final int NUM_BALLS_TO_TEST = 5; 041 private static final int BOX_CAPACITY = NUM_BALLS_TO_TEST - 1; 042 private static final double BALL_UNIT_VOLUME = 10.0; 043 private static final double JUNIT_DOUBLE_DELTA = 0.0001; 044 private static final int TRIES_FOR_BALLS_TEST = 3; 045 046 @BeforeClass 047 public static void setupBeforeTests() throws Exception { 048 assertThat("Test case error, you must test at least 1 Ball!!", 049 NUM_BALLS_TO_TEST, greaterThan(0)); 050 assertThat("This test case is set up assuming that the box cannot contain all the balls, please check and change parameters!", 051 NUM_BALLS_TO_TEST, greaterThan(BOX_CAPACITY)); 052 053 double box_volume = 0; 054 055 // Let's create the balls we need. 056 b = new Ball[NUM_BALLS_TO_TEST]; 057 for (int i=0; i<NUM_BALLS_TO_TEST; i++) { 058 if (i<BOX_CAPACITY) { 059 box_volume += (i+1)*BALL_UNIT_VOLUME; 060 } 061 b[i] = new Ball((i+1)*BALL_UNIT_VOLUME); 062 } 063 064 // Now, we create the box once we figure out how big a box we 065 // need. 066 box = new Box(box_volume); 067 068 } 069 070 /** Test to check that Box.add(Ball) is implemented correctly */ 071 @Test 072 public void testAdd() { 073 box.clear(); 074 for (int i=0; i<BOX_CAPACITY; i++) { 075 assertTrue("Box.add(Ball) failed to add a new Ball!", 076 box.add(b[i])); 077 assertFalse("Box.add(Ball) seems to allow the same Ball to be added twice!", 078 box.add(b[i])); 079 assertTrue("Box does not contain a ball after it is supposed to have been added!", 080 box.contains(b[i])); 081 } 082 for (int i=BOX_CAPACITY; i<NUM_BALLS_TO_TEST; i++) { 083 assertFalse("Box.add(Ball) allows a Ball to be added even though it is already full!", 084 box.add(b[i])); 085 } 086 } 087 088 /** Test to check that Box.getBallsFromSmallest() is implemented correctly */ 089 @Test 090 public void testGetBalls() { 091 Random rnd = new Random(); 092 093 for (int k=0; k<TRIES_FOR_BALLS_TEST; k++) { 094 095 box.clear(); 096 097 // Let's put all the balls into a list. 098 LinkedList<Ball> list = new LinkedList<Ball>(); 099 for (int i=0; i<BOX_CAPACITY; i++) { 100 list.add(b[i]); 101 } 102 103 // First we add the balls to the box in some random order. 104 for (int i=0; i<BOX_CAPACITY; i++) { 105 box.add(list.remove(rnd.nextInt(list.size()))); 106 } 107 108 int contentsSize = box.size(); 109 // Next we call the iterator and check that the balls come out 110 // in the correct order. 111 Iterator<Ball> it = box.getBallsFromSmallest(); 112 int count = 0; 113 while (it.hasNext() && count < BOX_CAPACITY) { 114 Ball ball = it.next(); 115 assertEquals("Balls are not returned by Box.getBallsFromSmallest() iterator in the correct order", 116 b[count], ball); 117 if (b[count] != ball) { 118 break; 119 } 120 count++; 121 } 122 assertEquals("Box.getBallsFromSmallest() did not return all the balls", 123 BOX_CAPACITY, count); 124 assertEquals("Number of balls in box was modified", 125 contentsSize, box.size()); 126 } 127 } 128 129 /** 130 * Test to check that Box.remove(Ball) is implemented 131 * correctly. Depending on how <code>getBallsFromSmallest()</code> 132 * is implemented, remove() might have to be overridden and this 133 * test helps ensure that remove() is not broken in the process. 134 */ 135 @Test 136 public void testRemove() { 137 box.clear(); 138 assertFalse("Box.remove(Ball) should fail because box is empty, but it didn't!", 139 box.remove(b[0])); 140 for (int i=0; i<BOX_CAPACITY; i++) { 141 box.clear(); 142 for (int j=0; j<i; j++) { 143 box.add(b[j]); 144 } 145 for (int j=0; j<i; j++) { 146 assertTrue("Box.remove(Ball) failed to remove a Ball that is supposed to be inside", 147 box.remove(b[j])); 148 assertFalse("Box still contains a ball after it is supposed to have been removed!", 149 box.contains(b[j])); 150 } 151 for (int j=i; j<NUM_BALLS_TO_TEST; j++) { 152 assertFalse("Box.remove(Ball) did not fail for a Ball that is not inside", 153 box.remove(b[j])); 154 } 155 } 156 } 157 158 159 /** Test to check that Box.clear() is implemented correctly */ 160 @Test 161 public void testClear() { 162 box.clear(); 163 assertEquals("Box is not empty after being cleared!", 0, box.size()); 164 box.add(b[0]); 165 box.clear(); 166 assertEquals("Box is not empty after being cleared!", 0, box.size()); 167 } 168 169 /** Test to check that check that we can put a Ball into a Box */ 170 @Test 171 public void testVolume() { 172 box.clear(); 173 assertEquals("Volume of empty Box is not zero!", 174 0, box.getVolume(), JUNIT_DOUBLE_DELTA); 175 for (int i=0; i<BOX_CAPACITY; i++) { 176 box.add(b[i]); 177 assertEquals("Volume of Box with one ball", 178 (i+1)*(i+2)*BALL_UNIT_VOLUME/2, box.getVolume(), 179 JUNIT_DOUBLE_DELTA); 180 } 181 } 182 183 /** Test to check that size() returns the correct number. */ 184 @Test 185 public void testSize() { 186 box.clear(); 187 assertEquals("size() of empty Box is not zero!", 0, box.size()); 188 for (int i=0; i<BOX_CAPACITY; i++) { 189 box.add(b[i]); 190 assertEquals("size() of Box with "+(i+1)+" ball(s)", 191 i+1, box.size()); 192 } 193 } 194 195 /** Test to check that size() returns the correct number. */ 196 @Test 197 public void testContains() { 198 box.clear(); 199 for (int i=0; i<BOX_CAPACITY; i++) { 200 assertFalse("Empty Box seems to contain a ball!", 201 box.contains(b[i])); 202 } 203 for (int i=0; i<BOX_CAPACITY; i++) { 204 box.add(b[i]); 205 assertTrue("Box does not contain a Ball that is supposed to be inside!", 206 box.contains(b[i])); 207 for (int j=i+1; j<BOX_CAPACITY; j++) { 208 assertFalse("Box seems to contain a Ball that is not inside!", 209 box.contains(b[j])); 210 } 211 } 212 } 213 214 /** 215 * Test to check that iterator() is implemented correctly. 216 */ 217 @Test 218 public void testIterator() { 219 Set<Ball> allBalls = new HashSet<Ball>(); 220 Set<Ball> seenBalls = new HashSet<Ball>(); 221 box.clear(); 222 for (Ball aBall : b) { 223 box.add(aBall); 224 allBalls.add(aBall); 225 } 226 int i = 0; 227 for (Ball aBall : box) { 228 assertTrue("Iterator returned a ball that isn't in the container: " + aBall, 229 allBalls.contains(aBall)); 230 assertFalse("Iterator returned the same ball twice: " + aBall, 231 seenBalls.contains(aBall)); 232 seenBalls.add(aBall); 233 i++; 234 } 235 assertEquals("BallContainer iterator did not return enough items!", 236 b.length-1, i); 237 } 238}