1 /*
2  * Copyright (C) 2009 The Guava Authors
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.google.common.collect;
18 
19 import static com.google.common.truth.Truth.assertThat;
20 import static java.util.Arrays.asList;
21 
22 import com.google.common.annotations.GwtCompatible;
23 import com.google.common.annotations.GwtIncompatible;
24 import com.google.common.base.Objects;
25 import com.google.common.collect.Table.Cell;
26 import com.google.common.testing.EqualsTester;
27 import com.google.common.testing.NullPointerTester;
28 import com.google.common.testing.SerializableTester;
29 
30 import java.util.Arrays;
31 import java.util.Map;
32 
33 /**
34  * Test cases for {@link ArrayTable}.
35  *
36  * @author Jared Levy
37  */
38 @GwtCompatible(emulated = true)
39 public class ArrayTableTest extends AbstractTableTest {
40 
create( Object... data)41   @Override protected ArrayTable<String, Integer, Character> create(
42       Object... data) {
43     // TODO: Specify different numbers of rows and columns, to detect problems
44     // that arise when the wrong size is used.
45     ArrayTable<String, Integer, Character> table =
46         ArrayTable.create(asList("foo", "bar", "cat"), asList(1, 2, 3));
47     populate(table, data);
48     return table;
49   }
50 
assertSize(int expectedSize)51   @Override protected void assertSize(int expectedSize) {
52     assertEquals(9, table.size());
53   }
54 
supportsRemove()55   @Override protected boolean supportsRemove() {
56     return false;
57   }
58 
supportsNullValues()59   @Override protected boolean supportsNullValues() {
60     return true;
61   }
62 
63   // Overriding tests of behavior that differs for ArrayTable.
64 
testContains()65   @Override public void testContains() {
66     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
67     assertTrue(table.contains("foo", 1));
68     assertTrue(table.contains("bar", 1));
69     assertTrue(table.contains("foo", 3));
70     assertTrue(table.contains("foo", 2));
71     assertTrue(table.contains("bar", 3));
72     assertTrue(table.contains("cat", 1));
73     assertFalse(table.contains("foo", -1));
74     assertFalse(table.contains("bad", 1));
75     assertFalse(table.contains("bad", -1));
76     assertFalse(table.contains("foo", null));
77     assertFalse(table.contains(null, 1));
78     assertFalse(table.contains(null, null));
79   }
80 
testContainsRow()81   @Override public void testContainsRow() {
82     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
83     assertTrue(table.containsRow("foo"));
84     assertTrue(table.containsRow("bar"));
85     assertTrue(table.containsRow("cat"));
86     assertFalse(table.containsRow("bad"));
87     assertFalse(table.containsRow(null));
88   }
89 
testContainsColumn()90   @Override public void testContainsColumn() {
91     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
92     assertTrue(table.containsColumn(1));
93     assertTrue(table.containsColumn(3));
94     assertTrue(table.containsColumn(2));
95     assertFalse(table.containsColumn(-1));
96     assertFalse(table.containsColumn(null));
97   }
98 
testContainsValue()99   @Override public void testContainsValue() {
100     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
101     assertTrue(table.containsValue('a'));
102     assertTrue(table.containsValue('b'));
103     assertTrue(table.containsValue('c'));
104     assertFalse(table.containsValue('x'));
105     assertTrue(table.containsValue(null));
106   }
107 
testIsEmpty()108   @Override public void testIsEmpty() {
109     assertFalse(table.isEmpty());
110     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
111     assertFalse(table.isEmpty());
112   }
113 
testEquals()114   @Override public void testEquals() {
115     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
116     Table<String, Integer, Character> hashCopy = HashBasedTable.create();
117     hashCopy.put("foo", 1, 'a');
118     hashCopy.put("bar", 1, 'b');
119     hashCopy.put("foo", 3, 'c');
120     Table<String, Integer, Character> reordered
121         = create("foo", 3, 'c', "foo", 1, 'a', "bar", 1, 'b');
122     Table<String, Integer, Character> smaller
123         = create("foo", 1, 'a', "bar", 1, 'b');
124     Table<String, Integer, Character> swapOuter
125         = create("bar", 1, 'a', "foo", 1, 'b', "bar", 3, 'c');
126     Table<String, Integer, Character> swapValues
127         = create("foo", 1, 'c', "bar", 1, 'b', "foo", 3, 'a');
128 
129     new EqualsTester()
130         .addEqualityGroup(table, reordered)
131         .addEqualityGroup(hashCopy)
132         .addEqualityGroup(smaller)
133         .addEqualityGroup(swapOuter)
134         .addEqualityGroup(swapValues)
135         .testEquals();
136   }
137 
testHashCode()138   @Override public void testHashCode() {
139     table = ArrayTable.create(asList("foo", "bar"), asList(1, 3));
140     table.put("foo", 1, 'a');
141     table.put("bar", 1, 'b');
142     table.put("foo", 3, 'c');
143     int expected = Objects.hashCode("foo", 1, 'a')
144         + Objects.hashCode("bar", 1, 'b')
145         + Objects.hashCode("foo", 3, 'c')
146         + Objects.hashCode("bar", 3, 0);
147     assertEquals(expected, table.hashCode());
148   }
149 
testRow()150   @Override public void testRow() {
151     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
152     Map<Integer, Character> expected = Maps.newHashMap();
153     expected.put(1, 'a');
154     expected.put(3, 'c');
155     expected.put(2, null);
156     assertEquals(expected, table.row("foo"));
157   }
158 
testColumn()159   @Override public void testColumn() {
160     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
161     Map<String, Character> expected = Maps.newHashMap();
162     expected.put("foo", 'a');
163     expected.put("bar", 'b');
164     expected.put("cat", null);
165     assertEquals(expected, table.column(1));
166   }
167 
testToStringSize1()168   @Override public void testToStringSize1() {
169     table = ArrayTable.create(ImmutableList.of("foo"), ImmutableList.of(1));
170     table.put("foo", 1, 'a');
171     assertEquals("{foo={1=a}}", table.toString());
172   }
173 
testCreateDuplicateRows()174   public void testCreateDuplicateRows() {
175     try {
176       ArrayTable.create(asList("foo", "bar", "foo"), asList(1, 2, 3));
177       fail();
178     } catch (IllegalArgumentException expected) {}
179   }
180 
testCreateDuplicateColumns()181   public void testCreateDuplicateColumns() {
182     try {
183       ArrayTable.create(asList("foo", "bar"), asList(1, 2, 3, 2));
184       fail();
185     } catch (IllegalArgumentException expected) {}
186   }
187 
testCreateEmptyRows()188   public void testCreateEmptyRows() {
189     try {
190       ArrayTable.create(Arrays.<String>asList(), asList(1, 2, 3));
191       fail();
192     } catch (IllegalArgumentException expected) {}
193   }
194 
testCreateEmptyColumns()195   public void testCreateEmptyColumns() {
196     try {
197       ArrayTable.create(asList("foo", "bar"), Arrays.<Integer>asList());
198       fail();
199     } catch (IllegalArgumentException expected) {}
200   }
201 
testCreateCopyArrayTable()202   public void testCreateCopyArrayTable() {
203     Table<String, Integer, Character> original
204         = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
205     Table<String, Integer, Character> copy = ArrayTable.create(original);
206     assertEquals(original, copy);
207     original.put("foo", 1, 'd');
208     assertEquals((Character) 'd', original.get("foo", 1));
209     assertEquals((Character) 'a', copy.get("foo", 1));
210     assertEquals(copy.rowKeySet(), original.rowKeySet());
211     assertEquals(copy.columnKeySet(), original.columnKeySet());
212   }
213 
testCreateCopyHashBasedTable()214   public void testCreateCopyHashBasedTable() {
215     Table<String, Integer, Character> original = HashBasedTable.create();
216     original.put("foo", 1, 'a');
217     original.put("bar", 1, 'b');
218     original.put("foo", 3, 'c');
219     Table<String, Integer, Character> copy = ArrayTable.create(original);
220     assertEquals(4, copy.size());
221     assertEquals((Character) 'a', copy.get("foo", 1));
222     assertEquals((Character) 'b', copy.get("bar", 1));
223     assertEquals((Character) 'c', copy.get("foo", 3));
224     assertNull(copy.get("bar", 3));
225     original.put("foo", 1, 'd');
226     assertEquals((Character) 'd', original.get("foo", 1));
227     assertEquals((Character) 'a', copy.get("foo", 1));
228     assertEquals(copy.rowKeySet(), ImmutableSet.of("foo", "bar"));
229     assertEquals(copy.columnKeySet(), ImmutableSet.of(1, 3));
230   }
231 
testCreateCopyEmptyTable()232   public void testCreateCopyEmptyTable() {
233     Table<String, Integer, Character> original = HashBasedTable.create();
234     try {
235       ArrayTable.create(original);
236       fail();
237     } catch (IllegalArgumentException expected) {}
238   }
239 
testSerialization()240   public void testSerialization() {
241     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
242     SerializableTester.reserializeAndAssert(table);
243   }
244 
245   @GwtIncompatible("reflection")
testNullPointerStatic()246   public void testNullPointerStatic() {
247     new NullPointerTester().testAllPublicStaticMethods(ArrayTable.class);
248   }
249 
testToString_ordered()250   public void testToString_ordered() {
251     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
252     assertEquals("{foo={1=a, 2=null, 3=c}, " +
253         "bar={1=b, 2=null, 3=null}, " +
254         "cat={1=null, 2=null, 3=null}}",
255         table.toString());
256     assertEquals("{foo={1=a, 2=null, 3=c}, " +
257         "bar={1=b, 2=null, 3=null}, " +
258         "cat={1=null, 2=null, 3=null}}",
259         table.rowMap().toString());
260   }
261 
testCellSetToString_ordered()262   public void testCellSetToString_ordered() {
263     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
264     assertEquals("[(foo,1)=a, (foo,2)=null, (foo,3)=c, " +
265         "(bar,1)=b, (bar,2)=null, (bar,3)=null, " +
266         "(cat,1)=null, (cat,2)=null, (cat,3)=null]",
267         table.cellSet().toString());
268   }
269 
testRowKeySetToString_ordered()270   public void testRowKeySetToString_ordered() {
271     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
272     assertEquals("[foo, bar, cat]", table.rowKeySet().toString());
273   }
274 
testColumnKeySetToString_ordered()275   public void testColumnKeySetToString_ordered() {
276     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
277     assertEquals("[1, 2, 3]", table.columnKeySet().toString());
278   }
279 
testValuesToString_ordered()280   public void testValuesToString_ordered() {
281     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
282     assertEquals("[a, null, c, b, null, null, null, null, null]",
283         table.values().toString());
284   }
285 
testRowKeyList()286   public void testRowKeyList() {
287     ArrayTable<String, Integer, Character> table
288         = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
289     assertThat(table.rowKeyList()).has().exactly("foo", "bar", "cat").inOrder();
290   }
291 
testColumnKeyList()292   public void testColumnKeyList() {
293     ArrayTable<String, Integer, Character> table
294         = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
295     assertThat(table.columnKeyList()).has().exactly(1, 2, 3).inOrder();
296   }
297 
testGetMissingKeys()298   public void testGetMissingKeys() {
299     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
300     assertNull(table.get("dog", 1));
301     assertNull(table.get("foo", 4));
302   }
303 
testAt()304   public void testAt() {
305     ArrayTable<String, Integer, Character> table
306         = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
307     assertEquals((Character) 'b', table.at(1, 0));
308     assertEquals((Character) 'c', table.at(0, 2));
309     assertNull(table.at(1, 2));
310     try {
311       table.at(1, 3);
312       fail();
313     } catch (IndexOutOfBoundsException expected) {}
314     try {
315       table.at(1, -1);
316       fail();
317     } catch (IndexOutOfBoundsException expected) {}
318     try {
319       table.at(3, 2);
320       fail();
321     } catch (IndexOutOfBoundsException expected) {}
322     try {
323       table.at(-1, 2);
324       fail();
325     } catch (IndexOutOfBoundsException expected) {}
326   }
327 
testSet()328   public void testSet() {
329     ArrayTable<String, Integer, Character> table
330         = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
331     assertEquals((Character) 'b', table.set(1, 0, 'd'));
332     assertEquals((Character) 'd', table.get("bar", 1));
333     assertNull(table.set(2, 0, 'e'));
334     assertEquals((Character) 'e', table.get("cat", 1));
335     assertEquals((Character) 'a', table.set(0, 0, null));
336     assertNull(table.get("foo", 1));
337     try {
338       table.set(1, 3, 'z');
339       fail();
340     } catch (IndexOutOfBoundsException expected) {}
341     try {
342       table.set(1, -1, 'z');
343       fail();
344     } catch (IndexOutOfBoundsException expected) {}
345     try {
346       table.set(3, 2, 'z');
347       fail();
348     } catch (IndexOutOfBoundsException expected) {}
349     try {
350       table.set(-1, 2, 'z');
351       fail();
352     } catch (IndexOutOfBoundsException expected) {}
353     assertFalse(table.containsValue('z'));
354   }
355 
testEraseAll()356   public void testEraseAll() {
357     ArrayTable<String, Integer, Character> table
358         = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
359     table.eraseAll();
360     assertEquals(9, table.size());
361     assertNull(table.get("bar", 1));
362     assertTrue(table.containsRow("foo"));
363     assertFalse(table.containsValue('a'));
364   }
365 
testPutIllegal()366   public void testPutIllegal() {
367     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
368     try {
369       table.put("dog", 1, 'd');
370       fail();
371     } catch (IllegalArgumentException expected) {
372       assertEquals("Row dog not in [foo, bar, cat]", expected.getMessage());
373     }
374     try {
375       table.put("foo", 4, 'd');
376       fail();
377     } catch (IllegalArgumentException expected) {
378       assertEquals("Column 4 not in [1, 2, 3]", expected.getMessage());
379     }
380     assertFalse(table.containsValue('d'));
381   }
382 
testErase()383   public void testErase() {
384     ArrayTable<String, Integer, Character> table
385         = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
386     assertEquals((Character) 'b', table.erase("bar", 1));
387     assertNull(table.get("bar", 1));
388     assertEquals(9, table.size());
389     assertNull(table.erase("bar", 1));
390     assertNull(table.erase("foo", 2));
391     assertNull(table.erase("dog", 1));
392     assertNull(table.erase("bar", 5));
393     assertNull(table.erase(null, 1));
394     assertNull(table.erase("bar", null));
395   }
396 
397   @GwtIncompatible("ArrayTable.toArray(Class)")
testToArray()398   public void testToArray() {
399     ArrayTable<String, Integer, Character> table
400         = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
401     Character[][] array = table.toArray(Character.class);
402     assertEquals(3, array.length);
403     assertThat(array[0]).asList().has().exactly('a', null, 'c').inOrder();
404     assertThat(array[1]).asList().has().exactly('b', null, null).inOrder();
405     assertThat(array[2]).asList().has().exactly(null, null, null).inOrder();
406     table.set(0, 2, 'd');
407     assertEquals((Character) 'c', array[0][2]);
408     array[0][2] = 'e';
409     assertEquals((Character) 'd', table.at(0, 2));
410   }
411 
testCellReflectsChanges()412   public void testCellReflectsChanges() {
413     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
414     Cell<String, Integer, Character> cell = table.cellSet().iterator().next();
415     assertEquals(Tables.immutableCell("foo", 1, 'a'), cell);
416     assertEquals((Character) 'a', table.put("foo", 1, 'd'));
417     assertEquals(Tables.immutableCell("foo", 1, 'd'), cell);
418   }
419 
testRowMissing()420   public void testRowMissing() {
421     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
422     Map<Integer, Character> row = table.row("dog");
423     assertTrue(row.isEmpty());
424     try {
425       row.put(1, 'd');
426       fail();
427     } catch (UnsupportedOperationException expected) {}
428   }
429 
testColumnMissing()430   public void testColumnMissing() {
431     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
432     Map<String, Character> column = table.column(4);
433     assertTrue(column.isEmpty());
434     try {
435       column.put("foo", 'd');
436       fail();
437     } catch (UnsupportedOperationException expected) {}
438   }
439 
testRowPutIllegal()440   public void testRowPutIllegal() {
441     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
442     Map<Integer, Character> map = table.row("foo");
443     try {
444       map.put(4, 'd');
445       fail();
446     } catch (IllegalArgumentException expected) {
447       assertEquals("Column 4 not in [1, 2, 3]", expected.getMessage());
448     }
449   }
450 
testColumnPutIllegal()451   public void testColumnPutIllegal() {
452     table = create("foo", 1, 'a', "bar", 1, 'b', "foo", 3, 'c');
453     Map<String, Character> map = table.column(3);
454     try {
455       map.put("dog", 'd');
456       fail();
457     } catch (IllegalArgumentException expected) {
458       assertEquals("Row dog not in [foo, bar, cat]", expected.getMessage());
459     }
460   }
461 
462   @GwtIncompatible("reflection")
testNulls()463   public void testNulls() {
464     new NullPointerTester().testAllPublicInstanceMethods(create());
465   }
466 
467   @GwtIncompatible("serialize")
testSerializable()468   public void testSerializable() {
469     SerializableTester.reserializeAndAssert(create());
470   }
471 }
472