BaseContactsProvider2Test.java revision 0be993f8ef0078b9825a5ffe6add08a6786d8dac
1d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov/*
2d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * Copyright (C) 2009 The Android Open Source Project
3d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov *
4d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * Licensed under the Apache License, Version 2.0 (the "License");
5d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * you may not use this file except in compliance with the License.
6d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * You may obtain a copy of the License at
7d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov *
8d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov *      http://www.apache.org/licenses/LICENSE-2.0
9d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov *
10d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * Unless required by applicable law or agreed to in writing, software
11d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * distributed under the License is distributed on an "AS IS" BASIS,
12d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * See the License for the specific language governing permissions and
14d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * limitations under the License.
15d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov */
1628f8857b1b46bde18b85c6d3c2a63ac44c3c2e1cEvan Millarpackage com.android.providers.contacts;
17d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
1828f8857b1b46bde18b85c6d3c2a63ac44c3c2e1cEvan Millarimport static com.android.providers.contacts.ContactsActor.PACKAGE_GREY;
19d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
20e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.accounts.Account;
21bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikovimport android.content.ContentProvider;
22b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikovimport android.content.ContentResolver;
23d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.content.ContentUris;
24d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.content.ContentValues;
25e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.content.Context;
269261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport android.content.Entity;
27a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport android.content.res.Resources;
28d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.database.Cursor;
29d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.net.Uri;
30bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikovimport android.provider.ContactsContract;
31d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.provider.ContactsContract.AggregationExceptions;
32e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.Contacts;
33d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.provider.ContactsContract.Data;
349261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport android.provider.ContactsContract.Groups;
35bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millarimport android.provider.ContactsContract.Presence;
36e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.RawContacts;
3789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikovimport android.provider.ContactsContract.Settings;
38a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Email;
399261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport android.provider.ContactsContract.CommonDataKinds.GroupMembership;
40e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Im;
41a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Nickname;
4201911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Organization;
43bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Phone;
44ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millarimport android.provider.ContactsContract.CommonDataKinds.Photo;
45e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.StructuredName;
464a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.StructuredPostal;
47d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.test.AndroidTestCase;
48d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.test.mock.MockContentResolver;
49d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.test.suitebuilder.annotation.LargeTest;
50b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikovimport android.util.Log;
51bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
52a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport java.io.ByteArrayOutputStream;
53a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport java.io.IOException;
54a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport java.io.InputStream;
559261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport java.util.ArrayList;
569261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport java.util.Arrays;
57e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport java.util.Comparator;
58ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikovimport java.util.Iterator;
5920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikovimport java.util.Map;
6020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikovimport java.util.Set;
61ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikovimport java.util.Map.Entry;
62d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
63d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov/**
64d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * A common superclass for {@link ContactsProvider2}-related tests.
65d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov */
66d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov@LargeTest
67d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovpublic abstract class BaseContactsProvider2Test extends AndroidTestCase {
68d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
69d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected static final String PACKAGE = "ContactsProvider2Test";
70d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
71bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected ContactsActor mActor;
72d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected MockContentResolver mResolver;
739261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected Account mAccount = new Account("account1", "account type1");
74e2579e029472f76b2dfda141444d775c67da0ec8Cynthia Wong    protected Account mAccountTwo = new Account("account2", "account type2");
759261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
762482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov    private byte[] mTestPhoto;
772482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov
789261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static Long NO_LONG = new Long(0);
799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static String NO_STRING = new String("");
809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static Account NO_ACCOUNT = new Account("a", "b");
81d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
82bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected Class<? extends ContentProvider> getProviderClass() {
83bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        return SynchronousContactsProvider2.class;
84bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    }
85bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
86bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected String getAuthority() {
87bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        return ContactsContract.AUTHORITY;
88bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    }
89bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
90d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    @Override
91d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected void setUp() throws Exception {
92d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        super.setUp();
93d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
94bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        mActor = new ContactsActor(getContext(), PACKAGE_GREY, getProviderClass(), getAuthority());
95d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        mResolver = mActor.resolver;
96445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov        if (mActor.provider instanceof SynchronousContactsProvider2) {
97445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov            ((SynchronousContactsProvider2) mActor.provider)
98445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov                    .getOpenHelper(mActor.context).wipeData();
99445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov        }
1009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
1019261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
102e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public Context getMockContext() {
103e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        return mActor.context;
104e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
105e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
106e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public void addAuthority(String authority) {
107e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        mActor.addAuthority(authority);
108e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
109e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
110e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public ContentProvider addProvider(Class<? extends ContentProvider> providerClass,
111e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov            String authority) throws Exception {
112e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        return mActor.addProvider(providerClass, authority);
113e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
114e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
1153d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov    public ContentProvider getProvider() {
1163d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov        return mActor.provider;
1173d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov    }
1183d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov
1199261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected Uri maybeAddAccountQueryParameters(Uri uri, Account account) {
1209261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == null) {
1219261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return uri;
1229261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
1239261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return uri.buildUpon()
124df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana                .appendQueryParameter(RawContacts.ACCOUNT_NAME, account.name)
125df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana                .appendQueryParameter(RawContacts.ACCOUNT_TYPE, account.type)
1269261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                .build();
127d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
128d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
129d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long createRawContact() {
130226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        return createRawContact(null);
131d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
132d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
133c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov    protected long createRawContactWithName() {
1343cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        return createRawContactWithName("John", "Doe");
1353cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov    }
1363cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov
1373cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov    protected long createRawContactWithName(String firstName, String lastName) {
138c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        long rawContactId = createRawContact(null);
1393cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        insertStructuredName(rawContactId, firstName, lastName);
140c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        return rawContactId;
141c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov    }
142c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov
143226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana    protected long createRawContact(Account account, String... extras) {
1449261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
145226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        for (int i = 0; i < extras.length; ) {
146226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana            values.put(extras[i], extras[i + 1]);
147226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana            i += 2;
148226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        }
1496cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov        final Uri uri = maybeAddAccountQueryParameters(RawContacts.CONTENT_URI, account);
1509261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Uri contactUri = mResolver.insert(uri, values);
1519261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return ContentUris.parseId(contactUri);
1529261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
1539261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
1549261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long createGroup(Account account, String sourceId, String title) {
1550be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov        return createGroup(account, sourceId, title, 1);
1560be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov    }
1570be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov
1580be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov    protected long createGroup(Account account, String sourceId, String title, int visible) {
1599261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
1609261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Groups.SOURCE_ID, sourceId);
1619261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Groups.TITLE, title);
1620be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov        values.put(Groups.GROUP_VISIBLE, visible);
1639261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        final Uri uri = maybeAddAccountQueryParameters(Groups.CONTENT_URI, account);
1649261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return ContentUris.parseId(mResolver.insert(uri, values));
1659261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
1669261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
16789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected void createSettings(Account account, String shouldSync, String ungroupedVisible) {
16889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        ContentValues values = new ContentValues();
16989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.ACCOUNT_NAME, account.name);
17089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.ACCOUNT_TYPE, account.type);
17189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.SHOULD_SYNC, shouldSync);
17289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.UNGROUPED_VISIBLE, ungroupedVisible);
17389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        mResolver.insert(Settings.CONTENT_URI, values);
17489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
17589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
1765ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertStructuredName(long rawContactId, String givenName, String familyName) {
177d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        ContentValues values = new ContentValues();
178d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        StringBuilder sb = new StringBuilder();
179d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (givenName != null) {
180d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(givenName);
181d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
182d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (givenName != null && familyName != null) {
183d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(" ");
184d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
185d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (familyName != null) {
186d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(familyName);
187d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
188d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.DISPLAY_NAME, sb.toString());
189d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.GIVEN_NAME, givenName);
190d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.FAMILY_NAME, familyName);
191d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
1925ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        return insertStructuredName(rawContactId, values);
1934097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov    }
1944097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
1955ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertStructuredName(long rawContactId, ContentValues values) {
1965ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
1974097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        values.put(Data.MIMETYPE, StructuredName.CONTENT_ITEM_TYPE);
198d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
199d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        return resultUri;
200d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
201d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
20201911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    protected Uri insertOrganization(long rawContactId, String organization) {
20301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        return insertOrganization(rawContactId, organization, false);
20401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    }
20501911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
20601911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    protected Uri insertOrganization(long rawContactId, String organization, boolean primary) {
20701911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        ContentValues values = new ContentValues();
20801911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
20901911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Data.MIMETYPE, Organization.CONTENT_ITEM_TYPE);
21001911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Organization.DATA, organization);
21101911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Organization.TYPE, Organization.TYPE_WORK);
21201911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        if (primary) {
21301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            values.put(Data.IS_PRIMARY, true);
21401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        }
21501911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
21601911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
21701911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        return resultUri;
21801911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    }
21901911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
2205ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber) {
22125abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        return insertPhoneNumber(rawContactId, phoneNumber, false);
22225abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    }
22325abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov
22425abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber, boolean primary) {
225bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        ContentValues values = new ContentValues();
2265ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
227bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        values.put(Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE);
228bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        values.put(Phone.NUMBER, phoneNumber);
2293cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Phone.TYPE, Phone.TYPE_HOME);
23025abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        if (primary) {
23125abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov            values.put(Data.IS_PRIMARY, true);
23225abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        }
233bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
234bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
235bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        return resultUri;
236bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov    }
237bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
2385ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email) {
23925abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        return insertEmail(rawContactId, email, false);
24025abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    }
24125abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov
24225abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email, boolean primary) {
243a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        ContentValues values = new ContentValues();
2445ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
245a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Data.MIMETYPE, Email.CONTENT_ITEM_TYPE);
246a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Email.DATA, email);
2473cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Email.TYPE, Email.TYPE_HOME);
24825abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        if (primary) {
24925abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov            values.put(Data.IS_PRIMARY, true);
25025abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        }
251a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
252a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
253a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        return resultUri;
254a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov    }
255a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
2565ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertNickname(long rawContactId, String nickname) {
257a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        ContentValues values = new ContentValues();
2585ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
259a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Data.MIMETYPE, Nickname.CONTENT_ITEM_TYPE);
260a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Nickname.NAME, nickname);
2613cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Nickname.TYPE, Nickname.TYPE_OTHER_NAME);
262a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
263a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
264a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        return resultUri;
265a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov    }
266a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
2674a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov    protected Uri insertPostalAddress(long rawContactId, String formattedAddress) {
2684a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        ContentValues values = new ContentValues();
2694a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
2704a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(Data.MIMETYPE, StructuredPostal.CONTENT_ITEM_TYPE);
2714a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(StructuredPostal.FORMATTED_ADDRESS, formattedAddress);
2724a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov
2734a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
2744a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        return resultUri;
2754a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov    }
2764a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov
2775ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertPhoto(long rawContactId) {
278ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        ContentValues values = new ContentValues();
2795ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
280ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        values.put(Data.MIMETYPE, Photo.CONTENT_ITEM_TYPE);
2812482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        values.put(Photo.PHOTO, loadTestPhoto());
282ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
283ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        return resultUri;
284ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
285ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
2865ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertGroupMembership(long rawContactId, String sourceId) {
2879261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
2885ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
2899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Data.MIMETYPE, GroupMembership.CONTENT_ITEM_TYPE);
2909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(GroupMembership.GROUP_SOURCE_ID, sourceId);
2919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return mResolver.insert(Data.CONTENT_URI, values);
2929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
2939261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
2945ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertGroupMembership(long rawContactId, Long groupId) {
2959261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
2965ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
2979261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Data.MIMETYPE, GroupMembership.CONTENT_ITEM_TYPE);
2989261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(GroupMembership.GROUP_ROW_ID, groupId);
2999261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return mResolver.insert(Data.CONTENT_URI, values);
3009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
3019261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
3024dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov    protected Uri insertPresence(int protocol, String customProtocol, String handle, int presence,
3034dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov            String status) {
304bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        ContentValues values = new ContentValues();
3054dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov        values.put(Presence.PROTOCOL, protocol);
3064dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov        values.put(Presence.CUSTOM_PROTOCOL, customProtocol);
307bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Presence.IM_HANDLE, handle);
308bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Presence.PRESENCE_STATUS, presence);
309e46667e641cd1c60998e1ccab4b60531d5b12ef7Dmitri Plotnikov        values.put(Presence.PRESENCE_CUSTOM_STATUS, status);
310bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
311bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        Uri resultUri = mResolver.insert(Presence.CONTENT_URI, values);
312bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        return resultUri;
313bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
314bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
3154dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov    protected Uri insertImHandle(long rawContactId, int protocol, String customProtocol,
3164dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov            String handle) {
317bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        ContentValues values = new ContentValues();
3185ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
319bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Data.MIMETYPE, Im.CONTENT_ITEM_TYPE);
320bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Im.PROTOCOL, protocol);
3214dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov        values.put(Im.CUSTOM_PROTOCOL, customProtocol);
322bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Im.DATA, handle);
3233cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Im.TYPE, Im.TYPE_HOME);
324bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
325bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
326bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        return resultUri;
327bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
328bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
3295ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected void setContactAccountName(long rawContactId, String accountName) {
330ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        ContentValues values = new ContentValues();
3316cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov        values.put(RawContacts.ACCOUNT_NAME, accountName);
332ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
333ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        mResolver.update(ContentUris.withAppendedId(
3345ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov                RawContacts.CONTENT_URI, rawContactId), values, null, null);
335ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
336ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
3370c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov    protected void setAggregationException(int type, long rawContactId1, long rawContactId2) {
338d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        ContentValues values = new ContentValues();
3390c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        values.put(AggregationExceptions.RAW_CONTACT_ID1, rawContactId1);
3400c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        values.put(AggregationExceptions.RAW_CONTACT_ID2, rawContactId2);
341d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(AggregationExceptions.TYPE, type);
3423d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov        assertEquals(1, mResolver.update(AggregationExceptions.CONTENT_URI, values, null, null));
343d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
344d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
345d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected Cursor queryRawContact(long rawContactId) {
3460c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
3470c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov                null, null, null, null);
348d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
349d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
350d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected Cursor queryContact(long contactId) {
351d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
352d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov                null, null, null, null);
353d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
354d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
355af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov    protected Cursor queryContact(long contactId, String[] projection) {
356af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
357af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov                projection, null, null, null);
358bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
359bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
360d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long queryContactId(long rawContactId) {
361d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryRawContact(rawContactId);
362d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertTrue(c.moveToFirst());
363d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId = c.getLong(c.getColumnIndex(RawContacts.CONTACT_ID));
364d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        c.close();
365d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        return contactId;
366d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
367d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
368d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long queryPhotoId(long contactId) {
369d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryContact(contactId);
370ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        assertTrue(c.moveToFirst());
371d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long photoId = c.getInt(c.getColumnIndex(Contacts.PHOTO_ID));
372ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        c.close();
373ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        return photoId;
374ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
375ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
376d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected String queryDisplayName(long contactId) {
377d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryContact(contactId);
378d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertTrue(c.moveToFirst());
379d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        String displayName = c.getString(c.getColumnIndex(Contacts.DISPLAY_NAME));
380d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        c.close();
381d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        return displayName;
382d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
383d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
384d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertAggregated(long rawContactId1, long rawContactId2) {
385285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov        forceAggregation();
386d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
387d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
388d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 == contactId2);
389bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov    }
390bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
391d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertAggregated(long rawContactId1, long rawContactId2,
392d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov            String expectedDisplayName) {
393285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov        forceAggregation();
394285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov
395d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
396d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
397d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 == contactId2);
398d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
399d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        String displayName = queryDisplayName(contactId1);
400d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertEquals(expectedDisplayName, displayName);
401d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
402d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
403d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertNotAggregated(long rawContactId1, long rawContactId2) {
404285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov        forceAggregation();
405285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov
406d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
407d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
408d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 != contactId2);
409d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
4104097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
4115ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected void assertStructuredName(long rawContactId, String prefix, String givenName,
4124097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov            String middleName, String familyName, String suffix) {
4136cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov        Uri uri =
4145ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov                Uri.withAppendedPath(ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
4156cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov                RawContacts.Data.CONTENT_DIRECTORY);
4164097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
4174097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        final String[] projection = new String[] {
4184097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                StructuredName.PREFIX, StructuredName.GIVEN_NAME, StructuredName.MIDDLE_NAME,
4194097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                StructuredName.FAMILY_NAME, StructuredName.SUFFIX
4204097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        };
4214097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
4224097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        Cursor c = mResolver.query(uri, projection, Data.MIMETYPE + "='"
4234097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                + StructuredName.CONTENT_ITEM_TYPE + "'", null, null);
4244097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
4254097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertTrue(c.moveToFirst());
4264097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(prefix, c.getString(0));
4274097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(givenName, c.getString(1));
4284097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(middleName, c.getString(2));
4294097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(familyName, c.getString(3));
4304097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(suffix, c.getString(4));
4314097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        c.close();
4324097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov    }
4339261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4349261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long assertSingleGroup(Long rowId, Account account, String sourceId, String title) {
4359261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(Groups.CONTENT_URI, null, null, null, null);
4369261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        try {
4379261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.moveToNext());
4389261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long actualRowId = assertGroup(c, rowId, account, sourceId, title);
4399261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertFalse(c.moveToNext());
4409261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return actualRowId;
4419261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } finally {
4429261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            c.close();
4439261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
4449261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
4459261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4465ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected long assertSingleGroupMembership(Long rowId, Long rawContactId, Long groupRowId,
4479261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String sourceId) {
4489261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(ContactsContract.Data.CONTENT_URI, null, null, null, null);
4499261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        try {
4509261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.moveToNext());
4515ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov            long actualRowId = assertGroupMembership(c, rowId, rawContactId, groupRowId, sourceId);
4529261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertFalse(c.moveToNext());
4539261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return actualRowId;
4549261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } finally {
4559261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            c.close();
4569261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
4579261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
4589261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4595ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected long assertGroupMembership(Cursor c, Long rowId, Long rawContactId, Long groupRowId,
4609261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String sourceId) {
4619261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, rowId, Data._ID);
4625ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        assertNullOrEquals(c, rawContactId, GroupMembership.RAW_CONTACT_ID);
4639261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, groupRowId, GroupMembership.GROUP_ROW_ID);
4649261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, sourceId, GroupMembership.GROUP_SOURCE_ID);
4659261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return c.getLong(c.getColumnIndexOrThrow("_id"));
4669261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
4679261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4689261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long assertGroup(Cursor c, Long rowId, Account account, String sourceId, String title) {
4699261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, rowId, Groups._ID);
4709261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, account);
4719261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, sourceId, Groups.SOURCE_ID);
4729261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, title, Groups.TITLE);
4739261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return c.getLong(c.getColumnIndexOrThrow("_id"));
4749261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
4759261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4769261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, Account account) {
4779261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == NO_ACCOUNT) {
4789261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return;
4799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
4809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == null) {
4819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.isNull(c.getColumnIndexOrThrow(Groups.ACCOUNT_NAME)));
4829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.isNull(c.getColumnIndexOrThrow(Groups.ACCOUNT_TYPE)));
4839261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } else {
484df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana            assertEquals(account.name, c.getString(c.getColumnIndexOrThrow(Groups.ACCOUNT_NAME)));
485df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana            assertEquals(account.type, c.getString(c.getColumnIndexOrThrow(Groups.ACCOUNT_TYPE)));
4869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
4879261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
4889261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, Long value, String columnName) {
4909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (value != NO_LONG) {
4919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (value == null) assertTrue(c.isNull(c.getColumnIndexOrThrow(columnName)));
4929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            else assertEquals((long) value, c.getLong(c.getColumnIndexOrThrow(columnName)));
4939261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
4949261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
4959261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4969261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, String value, String columnName) {
4979261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (value != NO_STRING) {
4989261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (value == null) assertTrue(c.isNull(c.getColumnIndexOrThrow(columnName)));
4999261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            else assertEquals(value, c.getString(c.getColumnIndexOrThrow(columnName)));
5009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5019261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5029261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5039261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void assertDataRow(ContentValues actual, String expectedMimetype,
5049261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            Object... expectedArguments) {
5059261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertEquals(actual.toString(), expectedMimetype, actual.getAsString(Data.MIMETYPE));
5069261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        for (int i = 0; i < expectedArguments.length; i += 2) {
5079261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String columnName = (String) expectedArguments[i];
5089261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            Object expectedValue = expectedArguments[i + 1];
5099261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue instanceof Uri) {
5109261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                expectedValue = ContentUris.parseId((Uri) expectedValue);
5119261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
5129261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue == null) {
5139261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                assertNull(actual.toString(), actual.get(columnName));
5149261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
5159261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue instanceof Long) {
516226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
517226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsLong(columnName));
5189261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else if (expectedValue instanceof Integer) {
519226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
520226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsInteger(columnName));
5219261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else if (expectedValue instanceof String) {
522226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
523226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsString(columnName));
5249261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else {
525226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
526226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.get(columnName));
5279261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
5289261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5299261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5309261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5319261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected static class IdComparator implements Comparator<ContentValues> {
5329261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        public int compare(ContentValues o1, ContentValues o2) {
5339261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long id1 = o1.getAsLong(ContactsContract.Data._ID);
5349261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long id2 = o2.getAsLong(ContactsContract.Data._ID);
5359261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (id1 == id2) return 0;
5369261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return (id1 < id2) ? -1 : 1;
5379261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5389261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5399261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5409261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected ContentValues[] asSortedContentValuesArray(
5419261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            ArrayList<Entity.NamedContentValues> subValues) {
5429261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues[] result = new ContentValues[subValues.size()];
5439261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        int i = 0;
5449261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        for (Entity.NamedContentValues subValue : subValues) {
5459261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            result[i] = subValue.values;
5469261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            i++;
5479261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5489261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Arrays.sort(result, new IdComparator());
5499261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return result;
5509261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5519261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5529261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void assertDirty(Uri uri, boolean state) {
5539261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(uri, new String[]{"dirty"}, null, null, null);
5549261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertTrue(c.moveToNext());
5559261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertEquals(state, c.getLong(0) != 0);
5569261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertFalse(c.moveToNext());
5573cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        c.close();
5589261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5599261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5609261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long getVersion(Uri uri) {
5619261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(uri, new String[]{"version"}, null, null, null);
5629261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertTrue(c.moveToNext());
5639261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        long version = c.getLong(0);
5649261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertFalse(c.moveToNext());
5653cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        c.close();
5669261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return version;
5679261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5689261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5699261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void clearDirty(Uri uri) {
5709261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
5719261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put("dirty", 0);
5729261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        mResolver.update(uri, values, null, null);
5739261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
57420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
5755870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    protected void storeValue(Uri contentUri, long id, String column, String value) {
5765870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        ContentValues values = new ContentValues();
5775870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        values.put(column, value);
5785870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov
5795870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        mResolver.update(ContentUris.withAppendedId(contentUri, id), values, null, null);
5805870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    }
5815870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov
5825870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    protected void assertStoredValue(Uri rowUri, String column, Object expectedValue) {
58320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        String value = getStoredValue(rowUri, column);
58401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        if (expectedValue == null) {
58501911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            assertNull("Column value " + column, value);
58601911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        } else {
58701911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            assertEquals("Column value " + column, String.valueOf(expectedValue), value);
58801911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        }
58920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
59020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
59189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected void assertStoredValue(Uri rowUri, String selection, String[] selectionArgs,
59289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            String column, Object expectedValue) {
59389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        String value = getStoredValue(rowUri, selection, selectionArgs, column);
59489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        if (expectedValue == null) {
59589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            assertNull("Column value " + column, value);
59689c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        } else {
59789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            assertEquals("Column value " + column, String.valueOf(expectedValue), value);
59889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        }
59989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
60089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
60120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected String getStoredValue(Uri rowUri, String column) {
60289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        return getStoredValue(rowUri, null, null, column);
60389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
60489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
60589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected String getStoredValue(Uri uri, String selection, String[] selectionArgs,
60689c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            String column) {
6078e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov        String value = null;
60889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        Cursor c = mResolver.query(uri, new String[] { column }, selection, selectionArgs, null);
60920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
6108e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov            if (c.moveToFirst()) {
6118e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov                value = c.getString(c.getColumnIndex(column));
6128e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov            }
61320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
61420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
61520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
61620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        return value;
61720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
61820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
61920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertStoredValues(Uri rowUri, ContentValues expectedValues) {
620bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov        assertStoredValues(rowUri, null, null, expectedValues);
621bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov    }
622bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov
623bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov    protected void assertStoredValues(Uri rowUri, String selection, String[] selectionArgs,
624bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov            ContentValues expectedValues) {
625bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov        Cursor c = mResolver.query(rowUri, null, selection, selectionArgs, null);
62620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
62720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
62820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.moveToFirst();
62920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertCursorValues(c, expectedValues);
63020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
63120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
63220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
63320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
63420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
635ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    protected void assertStoredValuesWithProjection(Uri rowUri, ContentValues expectedValues) {
636ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Cursor c = mResolver.query(rowUri, buildProjection(expectedValues), null, null, null);
637ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        try {
638ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
639ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            c.moveToFirst();
640ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            assertCursorValues(c, expectedValues);
641ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        } finally {
642ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            c.close();
643ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        }
644ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
645ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
64620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    /**
64720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * Constructs a selection (where clause) out of all supplied values, uses it
64820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * to query the provider and verifies that a single row is returned and it
64920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * has the same values as requested.
65020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     */
65120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertSelection(Uri uri, ContentValues values, String idColumn, long id) {
652ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        assertSelection(uri, values, idColumn, id, null);
653ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
654ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
655ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    public void assertSelectionWithProjection(Uri uri, ContentValues values, String idColumn,
656ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            long id) {
657ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        assertSelection(uri, values, idColumn, id, buildProjection(values));
658ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
659ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
660ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    private void assertSelection(Uri uri, ContentValues values, String idColumn, long id,
661ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            String[] projection) {
66220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        StringBuilder sb = new StringBuilder();
66320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        ArrayList<String> selectionArgs = new ArrayList<String>(values.size());
664c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        if (idColumn != null) {
665c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            sb.append(idColumn).append("=").append(id);
666c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        }
66720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        Set<Map.Entry<String, Object>> entries = values.valueSet();
66820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        for (Map.Entry<String, Object> entry : entries) {
66920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String column = entry.getKey();
67020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            Object value = entry.getValue();
671c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            if (sb.length() != 0) {
672c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov                sb.append(" AND ");
673c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            }
674c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            sb.append(column);
67520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            if (value == null) {
67620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                sb.append(" IS NULL");
67720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            } else {
67820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                sb.append("=?");
67920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                selectionArgs.add(String.valueOf(value));
68020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            }
68120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
68220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
683ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Cursor c = mResolver.query(uri, projection, sb.toString(), selectionArgs.toArray(new String[0]),
68420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                null);
68520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
68620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
68720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.moveToFirst();
68820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertCursorValues(c, values);
68920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
69020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
69120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
69220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
69320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
69420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertCursorValues(Cursor cursor, ContentValues expectedValues) {
69520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        Set<Map.Entry<String, Object>> entries = expectedValues.valueSet();
69620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        for (Map.Entry<String, Object> entry : entries) {
69720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String column = entry.getKey();
69820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            int index = cursor.getColumnIndex(column);
69920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertTrue("No such column: " + column, index != -1);
70020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            Object expectedValue = expectedValues.get(column);
70120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String value;
70220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            if (expectedValue instanceof byte[]) {
70320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                expectedValue = Hex.encodeHex((byte[])expectedValue, false);
70420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                value = Hex.encodeHex(cursor.getBlob(index), false);
70520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            } else {
70620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                expectedValue = expectedValues.getAsString(column);
70720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                value = cursor.getString(index);
70820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            }
70920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Column value " + column, expectedValue, value);
71020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
71120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
71233b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov
713ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    private String[] buildProjection(ContentValues values) {
714ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        String[] projection = new String[values.size()];
715ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Iterator<Entry<String, Object>> iter = values.valueSet().iterator();
716ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        for (int i = 0; i < projection.length; i++) {
717ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            projection[i] = iter.next().getKey();
718ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        }
719ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        return projection;
720ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
721ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
72233b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov    protected int getCount(Uri uri, String selection, String[] selectionArgs) {
72333b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        Cursor c = mResolver.query(uri, null, selection, selectionArgs, null);
72433b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        try {
72533b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov            return c.getCount();
72633b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        } finally {
72733b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov            c.close();
72833b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        }
72933b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov    }
730a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov
7312482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov    protected byte[] loadTestPhoto() {
7322482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        if (mTestPhoto == null) {
7332482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            final Resources resources = getContext().getResources();
7342482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            InputStream is = resources
7352482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                    .openRawResource(com.android.internal.R.drawable.ic_contact_picture);
7362482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            ByteArrayOutputStream os = new ByteArrayOutputStream();
7372482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            byte[] buffer = new byte[1000];
7382482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            int count;
7392482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            try {
7402482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                while ((count = is.read(buffer)) != -1) {
7412482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                    os.write(buffer, 0, count);
7422482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                }
7432482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            } catch (IOException e) {
7442482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                throw new RuntimeException(e);
7452482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            }
7462482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            mTestPhoto = os.toByteArray();
747a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov        }
7482482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        return mTestPhoto;
749a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov    }
750b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
751b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    public static void dump(ContentResolver resolver, boolean aggregatedOnly) {
752b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
753b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts._ID,
754b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts.DISPLAY_NAME
755b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
756b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String selection = null;
757b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        if (aggregatedOnly) {
758b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            selection = Contacts._ID
759b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                    + " IN (SELECT contact_id" +
760b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                    		" FROM raw_contacts GROUP BY contact_id HAVING count(*) > 1)";
761b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
762b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
763b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(Contacts.CONTENT_URI, projection, selection, null,
764b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts.DISPLAY_NAME);
765b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
766b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            long contactId = c.getLong(0);
767b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("Contact   ", String.format("%5d %s", contactId, c.getString(1)));
768b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            dumpRawContacts(resolver, contactId);
769b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("          ", ".");
770b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
771b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
772b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
773b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
774b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    private static void dumpRawContacts(ContentResolver resolver, long contactId) {
775b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
776b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                RawContacts._ID,
777b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
778b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(RawContacts.CONTENT_URI, projection, RawContacts.CONTACT_ID + "="
779b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                + contactId, null, null);
780b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
781b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            long rawContactId = c.getLong(0);
782b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("RawContact", String.format("      %-5d", rawContactId));
783b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            dumpData(resolver, rawContactId);
784b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
785b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
786b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
787b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
788b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    private static void dumpData(ContentResolver resolver, long rawContactId) {
789b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
790b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.MIMETYPE,
791b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA1,
792b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA2,
793b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA3,
794b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
795b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(Data.CONTENT_URI, projection, Data.RAW_CONTACT_ID + "="
796b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                + rawContactId, null, Data.MIMETYPE);
797b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
798b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            String mimetype = c.getString(0);
799b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            if (Photo.CONTENT_ITEM_TYPE.equals(mimetype)) {
800b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Log.i("Photo     ", "");
801b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            } else {
802b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                mimetype = mimetype.substring(mimetype.indexOf('/') + 1);
803b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Log.i("Data      ", String.format("            %-10s %s,%s,%s", mimetype,
804b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                        c.getString(1), c.getString(2), c.getString(3)));
805b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            }
806b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
807b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
808b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
809285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov
810285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov    protected void forceAggregation() {
811285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov        ((SynchronousContactsProvider2) mActor.provider).aggregate();
812285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov    }
81381d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov
81481d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov    protected void assertNetworkNotified(boolean expected) {
81581d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov        assertEquals(expected, ((SynchronousContactsProvider2)mActor.provider).isNetworkNotified());
81681d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov    }
817d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov}
818