blob: f0c4cbb462e82a643875efd536927498f0b200b4 [file] [log] [blame]
/*
* Copyright (C) 2013 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.dialer.dialpad;
import static com.android.dialer.dialpad.SmartDialCache.ContactNumber;
import com.android.dialer.dialpad.SmartDialTrie.Node;
import com.android.dialer.dialpad.SmartDialTrie.ParseInfo;
import android.test.suitebuilder.annotation.SmallTest;
import junit.framework.TestCase;
/**
* To run this test, use the command:
* adb shell am instrument -w -e class com.android.dialer.dialpad.SmartDialTrieTest /
* com.android.dialer.tests/android.test.InstrumentationTestRunner
*/
@SmallTest
public class SmartDialTrieTest extends TestCase{
public void testSize() {
final SmartDialTrie trie = new SmartDialTrie();
trie.put(new ContactNumber(0, "Jason", "0", "0", 1));
assertEquals(1, trie.size());
trie.put(new ContactNumber(1, "Mary", "0", "1", 2));
assertEquals(2, trie.size());
trie.put(new ContactNumber(2, "John", "0", "2", 3));
assertEquals(3, trie.size());
}
public void testPutForFullName() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber jasonsmith = new ContactNumber(0, "Jason Smith", "0", "0", 1);
final ContactNumber jasonsmitt = new ContactNumber(1, "Jason Smitt", "0", "1", 2);
trie.put(jasonsmith);
trie.put(jasonsmitt);
assertTrue(trie.getAllWithPrefix("5276676484").contains(jasonsmith));
assertFalse(trie.getAllWithPrefix("5276676484").contains(jasonsmitt));
assertFalse(trie.getAllWithPrefix("5276676488").contains(jasonsmith));
assertTrue(trie.getAllWithPrefix("5276676488").contains(jasonsmitt));
}
public void testPutForPartialName() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber maryjane = new ContactNumber(0, "Mary Jane", "0", "0", 1);
final ContactNumber sarahsmith = new ContactNumber(1, "Sarah Smith", "0", "1", 2);
final ContactNumber jasonsmitt = new ContactNumber(2, "Jason Smitt", "0", "2", 3);
trie.put(maryjane);
trie.put(sarahsmith);
trie.put(jasonsmitt);
// 6279 corresponds to mary = "Mary Jane" but not "Jason Smitt"
assertTrue(checkContains(trie, maryjane, "6279"));
assertFalse(checkContains(trie, jasonsmitt, "6279"));
// 72 corresponds to sa = "Sarah Smith" but not "Jason Smitt" or "Mary Jane"
assertFalse(checkContains(trie, maryjane, "72"));
assertTrue(checkContains(trie, sarahsmith, "72"));
assertFalse(checkContains(trie, jasonsmitt, "72"));
// 76 corresponds to sm = "Sarah Smith" and "Jason Smitt" but not "Mary Jane"
assertFalse(checkContains(trie, maryjane, "76"));
assertTrue(checkContains(trie, sarahsmith, "76"));
assertTrue(checkContains(trie, jasonsmitt, "76"));
}
public void testPutForNameTokens() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber jasonfwilliams = new ContactNumber(0, "Jason F. Williams", "0", "0", 1);
trie.put(jasonfwilliams);
// 527 corresponds to jas = "Jason"
assertTrue(checkContains(trie, jasonfwilliams, "527"));
// 945 corresponds to wil = "Wil"
assertTrue(checkContains(trie, jasonfwilliams, "945"));
// 66 doesn't match
assertFalse(checkContains(trie, jasonfwilliams, "66"));
}
public void testPutForInitialMatches() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber martinjuniorharry =
new ContactNumber(0, "Martin Jr Harry", "0", "0", 1);
trie.put(martinjuniorharry);
// 654 corresponds to mjh = "(M)artin (J)r (H)arry"
assertTrue(checkContains(trie, martinjuniorharry, "654"));
// The reverse (456) does not match (for now)
assertFalse(checkContains(trie, martinjuniorharry, "456"));
// 6542 corresponds to mjha = "(M)artin (J)r (Ha)rry"
assertTrue(checkContains(trie, martinjuniorharry, "6542"));
// 542 corresponds to jha = "Martin (J)r (Ha)rry"
assertTrue(checkContains(trie, martinjuniorharry, "542"));
// 642 corresponds to mha = "(M)artin Jr (Ha)rry"
assertTrue(checkContains(trie, martinjuniorharry, "642"));
// 6542779 (M)artin (J)r (Harry)
assertTrue(checkContains(trie, martinjuniorharry, "6542779"));
// 65742779 (M)artin (Jr) (Harry)
assertTrue(checkContains(trie, martinjuniorharry, "65742779"));
// 542779 Martin (J)r (Harry)
assertTrue(checkContains(trie, martinjuniorharry, "542779"));
// 547 doesn't match
assertFalse(checkContains(trie, martinjuniorharry, "547"));
// 655 doesn't match
assertFalse(checkContains(trie, martinjuniorharry, "655"));
// 653 doesn't match
assertFalse(checkContains(trie, martinjuniorharry, "653"));
// 6543 doesn't match
assertFalse(checkContains(trie, martinjuniorharry, "6543"));
// 7(2^3 -1) entries for the name, and 1 for the number
assertEquals(8, trie.numEntries());
}
public void testPutForInitialMatchesCombinations() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber alphabet = new ContactNumber(0, "abc def ghi jkl mno pqrs tuv wxyz",
"12345678", "1", 2);
trie.put(alphabet);
assertEquals(20, trie.numEntries());
// 8 name entries (abcdefghi..., defghi..., ...)
assertTrue(checkContains(trie, alphabet, "22233344455566677778889999"));
assertTrue(checkContains(trie, alphabet, "33344455566677778889999"));
assertTrue(checkContains(trie, alphabet, "44455566677778889999"));
assertTrue(checkContains(trie, alphabet, "55566677778889999"));
assertTrue(checkContains(trie, alphabet, "66677778889999"));
assertTrue(checkContains(trie, alphabet, "77778889999"));
assertTrue(checkContains(trie, alphabet, "8889999"));
assertTrue(checkContains(trie, alphabet, "9999"));
// 1 number entry
assertTrue(checkContains(trie, alphabet, "12345678"));
// 11 initial entries (adtw, adw, adt, ad, atw, at, aw, dt, dw, dtw, tw)
// 4c2(6) + 4c3(4) + 4c4(1)
assertTrue(checkContains(trie, alphabet, "2389999"));
assertTrue(checkContains(trie, alphabet, "239999"));
assertTrue(checkContains(trie, alphabet, "23888"));
assertTrue(checkContains(trie, alphabet, "2333"));
assertTrue(checkContains(trie, alphabet, "289999"));
assertTrue(checkContains(trie, alphabet, "2888"));
assertTrue(checkContains(trie, alphabet, "29999"));
assertTrue(checkContains(trie, alphabet, "3888"));
assertTrue(checkContains(trie, alphabet, "39999"));
assertTrue(checkContains(trie, alphabet, "389999"));
assertTrue(checkContains(trie, alphabet, "89999"));
}
public void testCheckLongToken() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber alphabet = new ContactNumber(0, " aaaa bbbb cccc dddd eeee ffff gggg" +
" hhhh iiii jjjj kkkk llll mmmm nnnn oooo pppp qqqq rrrr ssss tttt uuuu vvvv " +
" wwww xxxx yyyy zzzz", "1", "1", 2);
// Make sure the check to prevent overly long tokens from causing an OOM kicks in
trie.put(alphabet);
assertTrue(checkContains(trie, alphabet, "2222"));
// 26 name entries (aaaabbbbcccc...., bbbbccccdddd...., ccccdddd...)
// 1 number entry
// 11 initial entries 4c2(6) + 4c3(4) + 4c4(1)
assertEquals(38, trie.numEntries());
final ContactNumber alphabet2 = new ContactNumber(0, "aaaabbbbccccddddeeeeffffgggg" +
"hhhhiiiijjjjkkkkllllmmmmnnnnooooppppqqqqrrrrssssttttuuuuvvvvwwwwxxxxyyyyzzzz",
"1", "1", 2);
trie.put(alphabet2);
// added one name, and one number entry
assertEquals(40, trie.numEntries());
}
public void testParseInfo() {
final SmartDialTrie trie = new SmartDialTrie();
final String name = "Mcdonald Jamie-Cullum";
final ParseInfo info = trie.parseToIndexes(name, 2, 2);
// Make sure the dash is correctly converted to a separator character
for (int i = 0; i < name.length(); i++) {
// separators at position 8 and 12
if (i == 8 || i == 14) {
assertTrue(info.indexes[i] == -1);
} else {
assertFalse(info.indexes[i] == -1);
}
}
assertEquals(14, info.nthFirstTokenPos);
assertEquals(8, info.nthLastTokenPos);
final String name2 = "aaa bbb ccc ddd eee fff ggg hhh iii jjj kkk";
final ParseInfo info2 = trie.parseToIndexes(name2, 2, 2);
assertEquals(7, info2.nthFirstTokenPos);
assertEquals(35, info2.nthLastTokenPos);
final String name3 = "this is- a,test name";
final ParseInfo info3 = trie.parseToIndexes(name3, 3, 3);
assertEquals(11, info3.nthFirstTokenPos);
assertEquals(8, info3.nthLastTokenPos);
final String name4 = "M c-Donald James";
final ParseInfo info4 = trie.parseToIndexes(name4, 2, 3);
assertEquals(3, info4.nthFirstTokenPos);
assertEquals(1, info4.nthLastTokenPos);
final String name5 = " Aa'Bb c dddd e'e";
final ParseInfo info5 = trie.parseToIndexes(name5, 4, 4);
assertEquals(21, info5.nthFirstTokenPos);
assertEquals(8, info5.nthLastTokenPos);
}
public void testAccentedCharacters() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber reenee = new ContactNumber(0, "Reenée", "0", "0", 1);
final ContactNumber bronte = new ContactNumber(2, "Brontë", "0", "1", 2);
trie.put(reenee);
trie.put(bronte);
assertTrue(checkContains(trie, reenee, "733633"));
assertTrue(checkContains(trie, bronte, "276683"));
}
public void testNumbersInName() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber contact = new ContactNumber(0, "12345678", "0", "0", 1);
final ContactNumber teacher = new ContactNumber(1, "1st Grade Teacher", "0", "1", 2);
trie.put(contact);
trie.put(teacher);
assertTrue(checkContains(trie, contact, "12345678"));
// (1st Grade) Teacher
assertTrue(checkContains(trie, teacher, "17847233"));
// (1)st (G)rade (Tea)cher
assertTrue(checkContains(trie, teacher, "14832"));
}
public void testPutForNumbers() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber contactno1 = new ContactNumber(0, "James", "510-527-2357", "0", 1);
trie.put(contactno1);
final ContactNumber contactno2 = new ContactNumber(0, "James", "77212862357", "0", 1);
trie.put(contactno2);
final ContactNumber contactno3 = new ContactNumber(0, "James", "+13684976334", "0", 1);
trie.put(contactno3);
// all phone numbers belonging to the contact should correspond to it
assertTrue(checkContains(trie, contactno1, "510"));
assertFalse(checkContains(trie, contactno1, "511"));
assertTrue(checkContains(trie, contactno2, "77212862357"));
assertFalse(checkContains(trie, contactno2, "77212862356"));
assertTrue(checkContains(trie, contactno3, "1368"));
assertFalse(checkContains(trie, contactno3, "1367"));
}
public void testPutNumbersCountryCode() {
final SmartDialTrie trie = new SmartDialTrie();
final ContactNumber contactno1 = new ContactNumber(0, "James", "+13684976334", "0", 1);
trie.put(contactno1);
// all phone numbers belonging to the contact should correspond to it
assertTrue(checkContains(trie, contactno1, "1368"));
assertTrue(checkContains(trie, contactno1, "368497"));
assertFalse(checkContains(trie, contactno1, "2368497"));
final ContactNumber contactno2 = new ContactNumber(0, "Jason", "+65 9177-6930", "0", 1);
trie.put(contactno2);
assertTrue(checkContains(trie, contactno2, "6591776930"));
assertTrue(checkContains(trie, contactno2, "91776930"));
assertFalse(checkContains(trie, contactno2, "591776930"));
final ContactNumber contactno3 = new ContactNumber(0, "Mike", "+85212345678", "0", 1);
trie.put(contactno3);
assertTrue(checkContains(trie, contactno3, "85212345678"));
assertTrue(checkContains(trie, contactno3, "12345678"));
assertFalse(checkContains(trie, contactno2, "5212345678"));
// Invalid country code, don't try to parse it
final ContactNumber contactno4 = new ContactNumber(0, "Invalid", "+85112345678", "0", 1);
trie.put(contactno4);
assertTrue(checkContains(trie, contactno4, "85112345678"));
assertFalse(checkContains(trie, contactno4, "12345678"));
final ContactNumber contactno5 = new ContactNumber(0, "Invalid", "+852", "0", 1);
// Shouldn't crash
trie.put(contactno5);
}
// Tests special case handling for NANP numbers
public void testPutNumbersNANP() {
final SmartDialTrie trie = new SmartDialTrie(true /* formatNanp */);
// Unformatted number with 1 prefix
final ContactNumber contactno1 = new ContactNumber(0, "James", "16503337596", "0", 1);
trie.put(contactno1);
assertTrue(checkContains(trie, contactno1, "16503337596"));
assertTrue(checkContains(trie, contactno1, "6503337596"));
assertTrue(checkContains(trie, contactno1, "3337596"));
// Number with seperators
final ContactNumber contactno2 = new ContactNumber(0, "Michael", "5109921234", "0", 1);
trie.put(contactno2);
assertTrue(checkContains(trie, contactno2, "5109921234"));
assertTrue(checkContains(trie, contactno2, "9921234"));
// Number with area code only + separators
final ContactNumber contactno3 = new ContactNumber(0, "Jason", "(415)-123-4567", "0", 1);
trie.put(contactno3);
assertTrue(checkContains(trie, contactno3, "4151234567"));
assertTrue(checkContains(trie, contactno3, "1234567"));
// Number without +1 prefix but is a NANP number
final ContactNumber contactno4 = new ContactNumber(0, "Mike", "1 510-284-9170", "0", 1);
trie.put(contactno4);
assertTrue(checkContains(trie, contactno4, "15102849170"));
assertTrue(checkContains(trie, contactno4, "5102849170"));
assertTrue(checkContains(trie, contactno4, "2849170"));
// Invalid number(has 1 prefix, but is only 10 characters long)
final ContactNumber contactno5 = new ContactNumber(0, "Invalid", "1-415-123-123", "0", 1);
trie.put(contactno5);
// It should still be inserted as is
assertTrue(checkContains(trie, contactno5, "1415123123"));
// But the NANP special case handling should not work
assertFalse(checkContains(trie, contactno5, "415123123"));
assertFalse(checkContains(trie, contactno5, "123123"));
// Invalid number(only 9 characters long)
final ContactNumber contactno6 = new ContactNumber(0, "Invalid2", "415-123-123", "0", 1);
trie.put(contactno6);
// It should still be inserted as is
assertTrue(checkContains(trie, contactno6, "415123123"));
// But the NANP special case handling should not work
assertFalse(checkContains(trie, contactno6, "123123"));
// Number with +1 prefix and is a NANP number
final ContactNumber contactno7 = new ContactNumber(0, "Mike", "+1-510-284-9170", "0", 1);
trie.put(contactno7);
assertTrue(checkContains(trie, contactno7, "15102849170"));
assertTrue(checkContains(trie, contactno7, "5102849170"));
assertTrue(checkContains(trie, contactno7, "2849170"));
assertFalse(checkContains(trie, contactno7, "849170"));
assertFalse(checkContains(trie, contactno7, "10849170"));
// Number with +1 prefix but is an invalid NANP number
final ContactNumber contactno8 = new ContactNumber(0, "Invalid", "+1-510-284-917", "0", 1);
trie.put(contactno8);
assertTrue(checkContains(trie, contactno8, "1510284917"));
assertTrue(checkContains(trie, contactno8, "510284917"));
assertFalse(checkContains(trie, contactno8, "2849170"));
// Number with invalid country code prefix
final ContactNumber contactno9 = new ContactNumber(0, "Inv", "+857-510-284-9170", "0", 1);
trie.put(contactno9);
assertTrue(checkContains(trie, contactno9, "8575102849170"));
assertFalse(checkContains(trie, contactno9, "5102849170"));
assertFalse(checkContains(trie, contactno9, "2849170"));
// If user's region is determined to be not in North America, then the NANP number
// workarounds should not be applied
final SmartDialTrie trieNonNANP = new SmartDialTrie();
trieNonNANP.put(contactno3);
assertTrue(checkContains(trieNonNANP, contactno3, "4151234567"));
assertFalse(checkContains(trieNonNANP, contactno3, "1234567"));
trieNonNANP.put(contactno4);
assertTrue(checkContains(trieNonNANP, contactno4, "15102849170"));
assertFalse(checkContains(trieNonNANP, contactno4, "5102849170"));
assertFalse(checkContains(trieNonNANP, contactno4, "2849170"));
}
public void testNodeConstructor() {
final Node n = new Node();
// Node member variables should not be initialized by default at construction to reduce
// unnecessary memory usage
assertEquals(-1, n.getChildrenSize());
assertNull(n.getChild(5, false));
assertNull(n.getChild(0, false));
}
public void testNodeGetChild() {
final Node n = new Node();
// A node shouldn't contain children until getChild(index, true) is called
assertEquals(-1, n.getChildrenSize());
final Node child = n.getChild(1, true);
// A node should always have 10 children once the child array is created
assertEquals(10, n.getChildrenSize());
// getChild(index, true) should never return null
assertNotNull(child);
}
public void testNodeAddContact() {
final Node n = new Node();
assertNull(n.getContents());
final ContactNumber contact = new ContactNumber(0, "James", "510-527-2357", "0", 1);
final ContactNumber contactNotIn = new ContactNumber(2, "Jason Smitt", "0", "2", 3);
n.add(contact);
assertTrue(n.getContents().contains(contact));
assertFalse(n.getContents().contains(contactNotIn));
}
private boolean checkContains(SmartDialTrie trie, ContactNumber contact, CharSequence prefix) {
return trie.getAllWithPrefix(prefix).contains(contact);
}
}