BaseContactsProvider2Test.java revision 73f1f396c155b247b903d8f4111db17d3e13dc4d
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 */
168920a04b4a68ed6b548bcdef5ca8736dcf8b69b1Omari Stephens
1728f8857b1b46bde18b85c6d3c2a63ac44c3c2e1cEvan Millarpackage com.android.providers.contacts;
18d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
1928f8857b1b46bde18b85c6d3c2a63ac44c3c2e1cEvan Millarimport static com.android.providers.contacts.ContactsActor.PACKAGE_GREY;
20d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
21e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.accounts.Account;
22bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikovimport android.content.ContentProvider;
23b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikovimport android.content.ContentResolver;
24d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.content.ContentUris;
25d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.content.ContentValues;
26e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.content.Context;
279261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport android.content.Entity;
28a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport android.content.res.Resources;
29d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.database.Cursor;
30d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.net.Uri;
31bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikovimport android.provider.ContactsContract;
32d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.provider.ContactsContract.AggregationExceptions;
33e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.Contacts;
34d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.provider.ContactsContract.Data;
359261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport android.provider.ContactsContract.Groups;
36e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.RawContacts;
3789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikovimport android.provider.ContactsContract.Settings;
3882bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikovimport android.provider.ContactsContract.StatusUpdates;
39a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Email;
409261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport android.provider.ContactsContract.CommonDataKinds.GroupMembership;
41e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Im;
42a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Nickname;
4301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Organization;
44bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Phone;
45ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millarimport android.provider.ContactsContract.CommonDataKinds.Photo;
46e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.StructuredName;
474a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.StructuredPostal;
48d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.test.AndroidTestCase;
49d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.test.mock.MockContentResolver;
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 Plotnikovpublic abstract class BaseContactsProvider2Test extends AndroidTestCase {
67d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
68d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected static final String PACKAGE = "ContactsProvider2Test";
6973f1f396c155b247b903d8f4111db17d3e13dc4dDmitri Plotnikov    public static final String READ_ONLY_ACCOUNT_TYPE =
7073f1f396c155b247b903d8f4111db17d3e13dc4dDmitri Plotnikov            SynchronousContactsProvider2.READ_ONLY_ACCOUNT_TYPE;
71d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
72bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected ContactsActor mActor;
73d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected MockContentResolver mResolver;
749261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected Account mAccount = new Account("account1", "account type1");
75e2579e029472f76b2dfda141444d775c67da0ec8Cynthia Wong    protected Account mAccountTwo = new Account("account2", "account type2");
769261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
772482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov    private byte[] mTestPhoto;
782482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov
799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static Long NO_LONG = new Long(0);
809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static String NO_STRING = new String("");
819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static Account NO_ACCOUNT = new Account("a", "b");
82d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
83bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected Class<? extends ContentProvider> getProviderClass() {
84bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        return SynchronousContactsProvider2.class;
85bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    }
86bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
87bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected String getAuthority() {
88bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        return ContactsContract.AUTHORITY;
89bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    }
90bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
91d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    @Override
92d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected void setUp() throws Exception {
93d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        super.setUp();
94d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
95bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        mActor = new ContactsActor(getContext(), PACKAGE_GREY, getProviderClass(), getAuthority());
96d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        mResolver = mActor.resolver;
97445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov        if (mActor.provider instanceof SynchronousContactsProvider2) {
98445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov            ((SynchronousContactsProvider2) mActor.provider)
99b38ed2c5ffeb20efc677b4a9229db4a00603aa8dDmitri Plotnikov                    .getDatabaseHelper(mActor.context).wipeData();
100445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov        }
1019261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
1029261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
1030265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    @Override
1040265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    protected void tearDown() throws Exception {
1050265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov        if (mActor.provider instanceof SynchronousContactsProvider2) {
1060265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov            ((SynchronousContactsProvider2) mActor.provider)
1070265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov                    .getDatabaseHelper(mActor.context).close();
1080265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov        }
1090265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov        super.tearDown();
1100265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    }
1110265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov
112e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public Context getMockContext() {
113e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        return mActor.context;
114e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
115e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
116e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public void addAuthority(String authority) {
117e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        mActor.addAuthority(authority);
118e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
119e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
120e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public ContentProvider addProvider(Class<? extends ContentProvider> providerClass,
121e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov            String authority) throws Exception {
122e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        return mActor.addProvider(providerClass, authority);
123e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
124e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
1253d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov    public ContentProvider getProvider() {
1263d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov        return mActor.provider;
1273d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov    }
1283d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov
1299261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected Uri maybeAddAccountQueryParameters(Uri uri, Account account) {
1309261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == null) {
1319261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return uri;
1329261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
1339261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return uri.buildUpon()
134df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana                .appendQueryParameter(RawContacts.ACCOUNT_NAME, account.name)
135df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana                .appendQueryParameter(RawContacts.ACCOUNT_TYPE, account.type)
1369261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                .build();
137d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
138d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
139d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long createRawContact() {
140226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        return createRawContact(null);
141d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
142d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
143c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov    protected long createRawContactWithName() {
144e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName(null);
145e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
146e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
147e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected long createRawContactWithName(Account account) {
148e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName("John", "Doe", account);
1493cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov    }
1503cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov
1513cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov    protected long createRawContactWithName(String firstName, String lastName) {
152e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName(firstName, lastName, null);
153e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
154e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
155e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected long createRawContactWithName(String firstName, String lastName, Account account) {
156e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        long rawContactId = createRawContact(account);
1573cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        insertStructuredName(rawContactId, firstName, lastName);
158c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        return rawContactId;
159c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov    }
160c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov
161e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected Uri setCallerIsSyncAdapter(Uri uri, Account account) {
162e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        if (account == null) {
163e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey            return uri;
164e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        }
165e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        final Uri.Builder builder = uri.buildUpon();
166e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.RawContacts.ACCOUNT_NAME, account.name);
167e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.RawContacts.ACCOUNT_TYPE, account.type);
168e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true");
169e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return builder.build();
170e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
171e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
172226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana    protected long createRawContact(Account account, String... extras) {
1739261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
174226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        for (int i = 0; i < extras.length; ) {
175226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana            values.put(extras[i], extras[i + 1]);
176226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana            i += 2;
177226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        }
1786cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov        final Uri uri = maybeAddAccountQueryParameters(RawContacts.CONTENT_URI, account);
1799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Uri contactUri = mResolver.insert(uri, values);
1809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return ContentUris.parseId(contactUri);
1819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
1829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
1839261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long createGroup(Account account, String sourceId, String title) {
1840be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov        return createGroup(account, sourceId, title, 1);
1850be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov    }
1860be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov
1870be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov    protected long createGroup(Account account, String sourceId, String title, int visible) {
1889261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
1899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Groups.SOURCE_ID, sourceId);
1909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Groups.TITLE, title);
1910be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov        values.put(Groups.GROUP_VISIBLE, visible);
1929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        final Uri uri = maybeAddAccountQueryParameters(Groups.CONTENT_URI, account);
1939261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return ContentUris.parseId(mResolver.insert(uri, values));
1949261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
1959261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
19689c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected void createSettings(Account account, String shouldSync, String ungroupedVisible) {
19789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        ContentValues values = new ContentValues();
19889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.ACCOUNT_NAME, account.name);
19989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.ACCOUNT_TYPE, account.type);
20089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.SHOULD_SYNC, shouldSync);
20189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.UNGROUPED_VISIBLE, ungroupedVisible);
20289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        mResolver.insert(Settings.CONTENT_URI, values);
20389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
20489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
2055ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertStructuredName(long rawContactId, String givenName, String familyName) {
206d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        ContentValues values = new ContentValues();
207d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        StringBuilder sb = new StringBuilder();
208d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (givenName != null) {
209d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(givenName);
210d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
211d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (givenName != null && familyName != null) {
212d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(" ");
213d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
214d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (familyName != null) {
215d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(familyName);
216d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
217d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.DISPLAY_NAME, sb.toString());
218d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.GIVEN_NAME, givenName);
219d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.FAMILY_NAME, familyName);
220d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
2215ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        return insertStructuredName(rawContactId, values);
2224097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov    }
2234097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
2245ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertStructuredName(long rawContactId, ContentValues values) {
2255ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
2264097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        values.put(Data.MIMETYPE, StructuredName.CONTENT_ITEM_TYPE);
227d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
228d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        return resultUri;
229d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
230d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
231a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka    protected Uri insertOrganization(long rawContactId, ContentValues values) {
232a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka        return insertOrganization(rawContactId, values, false);
23301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    }
23401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
235a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka    protected Uri insertOrganization(long rawContactId, ContentValues values, boolean primary) {
23601911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
23701911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Data.MIMETYPE, Organization.CONTENT_ITEM_TYPE);
23801911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Organization.TYPE, Organization.TYPE_WORK);
23901911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        if (primary) {
2400265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
24101911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        }
24201911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
24301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
24401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        return resultUri;
24501911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    }
24601911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
2475ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber) {
24825abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        return insertPhoneNumber(rawContactId, phoneNumber, false);
24925abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    }
25025abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov
25125abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber, boolean primary) {
252bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        ContentValues values = new ContentValues();
2535ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
254bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        values.put(Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE);
255bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        values.put(Phone.NUMBER, phoneNumber);
2563cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Phone.TYPE, Phone.TYPE_HOME);
25725abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        if (primary) {
2580265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
25925abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        }
260bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
261bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
262bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        return resultUri;
263bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov    }
264bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
2655ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email) {
26625abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        return insertEmail(rawContactId, email, false);
26725abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    }
26825abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov
26925abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email, boolean primary) {
270916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        return insertEmail(rawContactId, email, primary, Email.TYPE_HOME, null);
271916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov    }
272916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov
273916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email, boolean primary, int type,
274916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov            String label) {
275a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        ContentValues values = new ContentValues();
2765ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
277a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Data.MIMETYPE, Email.CONTENT_ITEM_TYPE);
278a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Email.DATA, email);
279916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        values.put(Email.TYPE, type);
280916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        values.put(Email.LABEL, label);
28125abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        if (primary) {
282916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
28325abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        }
284a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
285a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
286a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        return resultUri;
287a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov    }
288a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
2895ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertNickname(long rawContactId, String nickname) {
290a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        ContentValues values = new ContentValues();
2915ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
292a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Data.MIMETYPE, Nickname.CONTENT_ITEM_TYPE);
293a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Nickname.NAME, nickname);
2943cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Nickname.TYPE, Nickname.TYPE_OTHER_NAME);
295a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
296a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
297a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        return resultUri;
298a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov    }
299a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
3004a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov    protected Uri insertPostalAddress(long rawContactId, String formattedAddress) {
3014a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        ContentValues values = new ContentValues();
3024a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3034a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(Data.MIMETYPE, StructuredPostal.CONTENT_ITEM_TYPE);
3044a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(StructuredPostal.FORMATTED_ADDRESS, formattedAddress);
3054a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov
3064a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
3074a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        return resultUri;
3084a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov    }
3094a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov
3105ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertPhoto(long rawContactId) {
311ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        ContentValues values = new ContentValues();
3125ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
313ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        values.put(Data.MIMETYPE, Photo.CONTENT_ITEM_TYPE);
3142482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        values.put(Photo.PHOTO, loadTestPhoto());
315ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
316ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        return resultUri;
317ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
318ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
3195ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertGroupMembership(long rawContactId, String sourceId) {
3209261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
3215ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3229261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Data.MIMETYPE, GroupMembership.CONTENT_ITEM_TYPE);
3239261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(GroupMembership.GROUP_SOURCE_ID, sourceId);
3249261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return mResolver.insert(Data.CONTENT_URI, values);
3259261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
3269261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
3275ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertGroupMembership(long rawContactId, Long groupId) {
3289261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
3295ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3309261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Data.MIMETYPE, GroupMembership.CONTENT_ITEM_TYPE);
3319261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(GroupMembership.GROUP_ROW_ID, groupId);
3329261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return mResolver.insert(Data.CONTENT_URI, values);
3339261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
3349261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
33582bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov    protected Uri insertStatusUpdate(int protocol, String customProtocol, String handle,
33682bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            int presence, String status) {
33782bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        return insertStatusUpdate(protocol, customProtocol, handle, presence, status, 0);
338a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov    }
339a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov
34082bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov    protected Uri insertStatusUpdate(int protocol, String customProtocol, String handle,
34182bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            int presence, String status, long timestamp) {
342bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        ContentValues values = new ContentValues();
34382bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.PROTOCOL, protocol);
34482bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.CUSTOM_PROTOCOL, customProtocol);
34582bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.IM_HANDLE, handle);
346a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (presence != 0) {
34782bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.PRESENCE, presence);
348a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
349a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (status != null) {
35082bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.STATUS, status);
351a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
352a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (timestamp != 0) {
35382bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.STATUS_TIMESTAMP, timestamp);
354a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
355bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
35682bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        Uri resultUri = mResolver.insert(StatusUpdates.CONTENT_URI, values);
357bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        return resultUri;
358bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
359bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
3604dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov    protected Uri insertImHandle(long rawContactId, int protocol, String customProtocol,
3614dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov            String handle) {
362bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        ContentValues values = new ContentValues();
3635ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
364bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Data.MIMETYPE, Im.CONTENT_ITEM_TYPE);
365bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Im.PROTOCOL, protocol);
3664dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov        values.put(Im.CUSTOM_PROTOCOL, customProtocol);
367bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Im.DATA, handle);
3683cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Im.TYPE, Im.TYPE_HOME);
369bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
370bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
371bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        return resultUri;
372bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
373bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
3745ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected void setContactAccountName(long rawContactId, String accountName) {
375ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        ContentValues values = new ContentValues();
3766cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov        values.put(RawContacts.ACCOUNT_NAME, accountName);
377ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
378ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        mResolver.update(ContentUris.withAppendedId(
3795ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov                RawContacts.CONTENT_URI, rawContactId), values, null, null);
380ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
381ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
3820c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov    protected void setAggregationException(int type, long rawContactId1, long rawContactId2) {
383d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        ContentValues values = new ContentValues();
3840c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        values.put(AggregationExceptions.RAW_CONTACT_ID1, rawContactId1);
3850c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        values.put(AggregationExceptions.RAW_CONTACT_ID2, rawContactId2);
386d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(AggregationExceptions.TYPE, type);
3873d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov        assertEquals(1, mResolver.update(AggregationExceptions.CONTENT_URI, values, null, null));
388d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
389d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
390d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected Cursor queryRawContact(long rawContactId) {
3910c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
3920c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov                null, null, null, null);
393d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
394d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
395d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected Cursor queryContact(long contactId) {
396d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
397d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov                null, null, null, null);
398d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
399d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
400af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov    protected Cursor queryContact(long contactId, String[] projection) {
401af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
402af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov                projection, null, null, null);
403bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
404bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
405d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long queryContactId(long rawContactId) {
406d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryRawContact(rawContactId);
407d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertTrue(c.moveToFirst());
408d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId = c.getLong(c.getColumnIndex(RawContacts.CONTACT_ID));
409d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        c.close();
410d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        return contactId;
411d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
412d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
413d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long queryPhotoId(long contactId) {
414d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryContact(contactId);
415ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        assertTrue(c.moveToFirst());
416d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long photoId = c.getInt(c.getColumnIndex(Contacts.PHOTO_ID));
417ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        c.close();
418ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        return photoId;
419ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
420ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
421d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected String queryDisplayName(long contactId) {
422d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryContact(contactId);
423d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertTrue(c.moveToFirst());
424d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        String displayName = c.getString(c.getColumnIndex(Contacts.DISPLAY_NAME));
425d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        c.close();
426d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        return displayName;
427d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
428d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
4292d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill    private String queryLookupKey(long contactId) {
4302d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        Cursor c = queryContact(contactId);
4312d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        assertTrue(c.moveToFirst());
4322d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        String lookupKey = c.getString(c.getColumnIndex(Contacts.LOOKUP_KEY));
4332d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        c.close();
4342d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        return lookupKey;
4352d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill    }
4362d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill
437d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertAggregated(long rawContactId1, long rawContactId2) {
438d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
439d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
440d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 == contactId2);
441bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov    }
442bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
443d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertAggregated(long rawContactId1, long rawContactId2,
444d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov            String expectedDisplayName) {
445d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
446d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
447d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 == contactId2);
448d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
449d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        String displayName = queryDisplayName(contactId1);
450d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertEquals(expectedDisplayName, displayName);
451d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
452d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
453d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertNotAggregated(long rawContactId1, long rawContactId2) {
454d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
455d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
456d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 != contactId2);
457d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
4584097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
4595ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected void assertStructuredName(long rawContactId, String prefix, String givenName,
4604097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov            String middleName, String familyName, String suffix) {
4616cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov        Uri uri =
4625ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov                Uri.withAppendedPath(ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
4636cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov                RawContacts.Data.CONTENT_DIRECTORY);
4644097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
4654097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        final String[] projection = new String[] {
4664097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                StructuredName.PREFIX, StructuredName.GIVEN_NAME, StructuredName.MIDDLE_NAME,
4674097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                StructuredName.FAMILY_NAME, StructuredName.SUFFIX
4684097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        };
4694097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
4704097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        Cursor c = mResolver.query(uri, projection, Data.MIMETYPE + "='"
4714097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                + StructuredName.CONTENT_ITEM_TYPE + "'", null, null);
4724097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
4734097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertTrue(c.moveToFirst());
4744097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(prefix, c.getString(0));
4754097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(givenName, c.getString(1));
4764097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(middleName, c.getString(2));
4774097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(familyName, c.getString(3));
4784097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(suffix, c.getString(4));
4794097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        c.close();
4804097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov    }
4819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long assertSingleGroup(Long rowId, Account account, String sourceId, String title) {
4839261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(Groups.CONTENT_URI, null, null, null, null);
4849261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        try {
4859261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.moveToNext());
4869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long actualRowId = assertGroup(c, rowId, account, sourceId, title);
4879261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertFalse(c.moveToNext());
4889261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return actualRowId;
4899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } finally {
4909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            c.close();
4919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
4929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
4939261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
4945ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected long assertSingleGroupMembership(Long rowId, Long rawContactId, Long groupRowId,
4959261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String sourceId) {
4969261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(ContactsContract.Data.CONTENT_URI, null, null, null, null);
4979261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        try {
4989261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.moveToNext());
4995ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov            long actualRowId = assertGroupMembership(c, rowId, rawContactId, groupRowId, sourceId);
5009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertFalse(c.moveToNext());
5019261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return actualRowId;
5029261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } finally {
5039261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            c.close();
5049261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5059261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5069261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5075ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected long assertGroupMembership(Cursor c, Long rowId, Long rawContactId, Long groupRowId,
5089261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String sourceId) {
5099261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, rowId, Data._ID);
5105ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        assertNullOrEquals(c, rawContactId, GroupMembership.RAW_CONTACT_ID);
5119261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, groupRowId, GroupMembership.GROUP_ROW_ID);
5129261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, sourceId, GroupMembership.GROUP_SOURCE_ID);
5139261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return c.getLong(c.getColumnIndexOrThrow("_id"));
5149261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5159261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5169261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long assertGroup(Cursor c, Long rowId, Account account, String sourceId, String title) {
5179261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, rowId, Groups._ID);
5189261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, account);
5199261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, sourceId, Groups.SOURCE_ID);
5209261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, title, Groups.TITLE);
5219261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return c.getLong(c.getColumnIndexOrThrow("_id"));
5229261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5239261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5249261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, Account account) {
5259261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == NO_ACCOUNT) {
5269261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return;
5279261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5289261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == null) {
5299261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.isNull(c.getColumnIndexOrThrow(Groups.ACCOUNT_NAME)));
5309261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.isNull(c.getColumnIndexOrThrow(Groups.ACCOUNT_TYPE)));
5319261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } else {
532df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana            assertEquals(account.name, c.getString(c.getColumnIndexOrThrow(Groups.ACCOUNT_NAME)));
533df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana            assertEquals(account.type, c.getString(c.getColumnIndexOrThrow(Groups.ACCOUNT_TYPE)));
5349261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5359261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5369261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5379261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, Long value, String columnName) {
5389261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (value != NO_LONG) {
5399261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (value == null) assertTrue(c.isNull(c.getColumnIndexOrThrow(columnName)));
5409261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            else assertEquals((long) value, c.getLong(c.getColumnIndexOrThrow(columnName)));
5419261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5429261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5439261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5449261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, String value, String columnName) {
5459261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (value != NO_STRING) {
5469261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (value == null) assertTrue(c.isNull(c.getColumnIndexOrThrow(columnName)));
5479261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            else assertEquals(value, c.getString(c.getColumnIndexOrThrow(columnName)));
5489261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5499261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5509261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5519261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void assertDataRow(ContentValues actual, String expectedMimetype,
5529261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            Object... expectedArguments) {
5539261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertEquals(actual.toString(), expectedMimetype, actual.getAsString(Data.MIMETYPE));
5549261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        for (int i = 0; i < expectedArguments.length; i += 2) {
5559261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String columnName = (String) expectedArguments[i];
5569261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            Object expectedValue = expectedArguments[i + 1];
5579261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue instanceof Uri) {
5589261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                expectedValue = ContentUris.parseId((Uri) expectedValue);
5599261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
5609261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue == null) {
5619261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                assertNull(actual.toString(), actual.get(columnName));
5629261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
5639261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue instanceof Long) {
564226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
565226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsLong(columnName));
5669261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else if (expectedValue instanceof Integer) {
567226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
568226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsInteger(columnName));
5699261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else if (expectedValue instanceof String) {
570226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
571226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsString(columnName));
5729261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else {
573226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
574226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.get(columnName));
5759261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
5769261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5779261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5789261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected static class IdComparator implements Comparator<ContentValues> {
5809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        public int compare(ContentValues o1, ContentValues o2) {
5819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long id1 = o1.getAsLong(ContactsContract.Data._ID);
5829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long id2 = o2.getAsLong(ContactsContract.Data._ID);
5839261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (id1 == id2) return 0;
5849261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return (id1 < id2) ? -1 : 1;
5859261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5879261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5889261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected ContentValues[] asSortedContentValuesArray(
5899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            ArrayList<Entity.NamedContentValues> subValues) {
5909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues[] result = new ContentValues[subValues.size()];
5919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        int i = 0;
5929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        for (Entity.NamedContentValues subValue : subValues) {
5939261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            result[i] = subValue.values;
5949261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            i++;
5959261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5969261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Arrays.sort(result, new IdComparator());
5979261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return result;
5989261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5999261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void assertDirty(Uri uri, boolean state) {
6019261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(uri, new String[]{"dirty"}, null, null, null);
6029261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertTrue(c.moveToNext());
6039261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertEquals(state, c.getLong(0) != 0);
6049261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertFalse(c.moveToNext());
6053cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        c.close();
6069261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6079261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6089261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long getVersion(Uri uri) {
6099261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(uri, new String[]{"version"}, null, null, null);
6109261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertTrue(c.moveToNext());
6119261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        long version = c.getLong(0);
6129261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertFalse(c.moveToNext());
6133cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        c.close();
6149261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return version;
6159261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6169261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6179261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void clearDirty(Uri uri) {
6189261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
6199261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put("dirty", 0);
6209261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        mResolver.update(uri, values, null, null);
6219261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
62220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
6235870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    protected void storeValue(Uri contentUri, long id, String column, String value) {
624f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        storeValue(ContentUris.withAppendedId(contentUri, id), column, value);
625f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
626f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
627f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, String column, String value) {
628f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        ContentValues values = new ContentValues();
629f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        values.put(column, value);
630f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
631f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        mResolver.update(contentUri, values, null, null);
632f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
633f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
634f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, long id, String column, long value) {
635f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        storeValue(ContentUris.withAppendedId(contentUri, id), column, value);
636f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
637f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
638f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, String column, long value) {
6395870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        ContentValues values = new ContentValues();
6405870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        values.put(column, value);
6415870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov
642f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        mResolver.update(contentUri, values, null, null);
6435870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    }
6445870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov
645f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov    protected void assertStoredValue(Uri contentUri, long id, String column, Object expectedValue) {
646f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov        assertStoredValue(ContentUris.withAppendedId(contentUri, id), column, expectedValue);
647f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov    }
648f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov
6495870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    protected void assertStoredValue(Uri rowUri, String column, Object expectedValue) {
65020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        String value = getStoredValue(rowUri, column);
65101911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        if (expectedValue == null) {
65201911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            assertNull("Column value " + column, value);
65301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        } else {
65401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            assertEquals("Column value " + column, String.valueOf(expectedValue), value);
65501911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        }
65620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
65720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
65889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected void assertStoredValue(Uri rowUri, String selection, String[] selectionArgs,
65989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            String column, Object expectedValue) {
66089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        String value = getStoredValue(rowUri, selection, selectionArgs, column);
66189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        if (expectedValue == null) {
66289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            assertNull("Column value " + column, value);
66389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        } else {
66489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            assertEquals("Column value " + column, String.valueOf(expectedValue), value);
66589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        }
66689c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
66789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
66820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected String getStoredValue(Uri rowUri, String column) {
66989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        return getStoredValue(rowUri, null, null, column);
67089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
67189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
67289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected String getStoredValue(Uri uri, String selection, String[] selectionArgs,
67389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            String column) {
6748e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov        String value = null;
67589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        Cursor c = mResolver.query(uri, new String[] { column }, selection, selectionArgs, null);
67620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
6778e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov            if (c.moveToFirst()) {
6788e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov                value = c.getString(c.getColumnIndex(column));
6798e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov            }
68020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
68120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
68220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
68320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        return value;
68420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
68520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
68620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertStoredValues(Uri rowUri, ContentValues expectedValues) {
687bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov        assertStoredValues(rowUri, null, null, expectedValues);
688bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov    }
689bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov
690bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov    protected void assertStoredValues(Uri rowUri, String selection, String[] selectionArgs,
691bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov            ContentValues expectedValues) {
692bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov        Cursor c = mResolver.query(rowUri, null, selection, selectionArgs, null);
69320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
69420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
69520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.moveToFirst();
69620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertCursorValues(c, expectedValues);
69720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
69820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
69920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
70020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
70120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
702ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    protected void assertStoredValuesWithProjection(Uri rowUri, ContentValues expectedValues) {
703ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Cursor c = mResolver.query(rowUri, buildProjection(expectedValues), null, null, null);
704ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        try {
705ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
706ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            c.moveToFirst();
707ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            assertCursorValues(c, expectedValues);
708ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        } finally {
709ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            c.close();
710ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        }
711ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
712ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
71320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    /**
71420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * Constructs a selection (where clause) out of all supplied values, uses it
71520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * to query the provider and verifies that a single row is returned and it
71620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * has the same values as requested.
71720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     */
71820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertSelection(Uri uri, ContentValues values, String idColumn, long id) {
719ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        assertSelection(uri, values, idColumn, id, null);
720ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
721ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
722ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    public void assertSelectionWithProjection(Uri uri, ContentValues values, String idColumn,
723ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            long id) {
724ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        assertSelection(uri, values, idColumn, id, buildProjection(values));
725ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
726ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
727ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    private void assertSelection(Uri uri, ContentValues values, String idColumn, long id,
728ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            String[] projection) {
72920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        StringBuilder sb = new StringBuilder();
73020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        ArrayList<String> selectionArgs = new ArrayList<String>(values.size());
731c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        if (idColumn != null) {
732c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            sb.append(idColumn).append("=").append(id);
733c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        }
73420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        Set<Map.Entry<String, Object>> entries = values.valueSet();
73520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        for (Map.Entry<String, Object> entry : entries) {
73620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String column = entry.getKey();
73720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            Object value = entry.getValue();
738c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            if (sb.length() != 0) {
739c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov                sb.append(" AND ");
740c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            }
741c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            sb.append(column);
74220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            if (value == null) {
74320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                sb.append(" IS NULL");
74420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            } else {
74520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                sb.append("=?");
74620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                selectionArgs.add(String.valueOf(value));
74720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            }
74820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
74920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
750ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Cursor c = mResolver.query(uri, projection, sb.toString(), selectionArgs.toArray(new String[0]),
75120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                null);
75220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
75320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
75420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.moveToFirst();
75520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertCursorValues(c, values);
75620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
75720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
75820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
75920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
76020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
76120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertCursorValues(Cursor cursor, ContentValues expectedValues) {
76220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        Set<Map.Entry<String, Object>> entries = expectedValues.valueSet();
76320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        for (Map.Entry<String, Object> entry : entries) {
76420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String column = entry.getKey();
76520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            int index = cursor.getColumnIndex(column);
76620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertTrue("No such column: " + column, index != -1);
76720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            Object expectedValue = expectedValues.get(column);
76820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String value;
76920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            if (expectedValue instanceof byte[]) {
77020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                expectedValue = Hex.encodeHex((byte[])expectedValue, false);
77120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                value = Hex.encodeHex(cursor.getBlob(index), false);
77220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            } else {
77320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                expectedValue = expectedValues.getAsString(column);
77420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                value = cursor.getString(index);
77520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            }
77620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Column value " + column, expectedValue, value);
77720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
77820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
77933b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov
780ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    private String[] buildProjection(ContentValues values) {
781ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        String[] projection = new String[values.size()];
782ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Iterator<Entry<String, Object>> iter = values.valueSet().iterator();
783ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        for (int i = 0; i < projection.length; i++) {
784ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            projection[i] = iter.next().getKey();
785ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        }
786ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        return projection;
787ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
788ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
78933b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov    protected int getCount(Uri uri, String selection, String[] selectionArgs) {
79033b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        Cursor c = mResolver.query(uri, null, selection, selectionArgs, null);
79133b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        try {
79233b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov            return c.getCount();
79333b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        } finally {
79433b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov            c.close();
79533b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        }
79633b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov    }
797a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov
7982482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov    protected byte[] loadTestPhoto() {
7992482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        if (mTestPhoto == null) {
8002482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            final Resources resources = getContext().getResources();
8012482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            InputStream is = resources
8022482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                    .openRawResource(com.android.internal.R.drawable.ic_contact_picture);
8032482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            ByteArrayOutputStream os = new ByteArrayOutputStream();
8042482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            byte[] buffer = new byte[1000];
8052482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            int count;
8062482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            try {
8072482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                while ((count = is.read(buffer)) != -1) {
8082482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                    os.write(buffer, 0, count);
8092482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                }
8102482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            } catch (IOException e) {
8112482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                throw new RuntimeException(e);
8122482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            }
8132482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            mTestPhoto = os.toByteArray();
814a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov        }
8152482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        return mTestPhoto;
816a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov    }
817b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
818b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    public static void dump(ContentResolver resolver, boolean aggregatedOnly) {
819b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
820b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts._ID,
821b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts.DISPLAY_NAME
822b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
823b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String selection = null;
824b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        if (aggregatedOnly) {
825b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            selection = Contacts._ID
826b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                    + " IN (SELECT contact_id" +
827b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                    		" FROM raw_contacts GROUP BY contact_id HAVING count(*) > 1)";
828b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
829b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
830b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(Contacts.CONTENT_URI, projection, selection, null,
831b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts.DISPLAY_NAME);
832b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
833b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            long contactId = c.getLong(0);
834b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("Contact   ", String.format("%5d %s", contactId, c.getString(1)));
835b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            dumpRawContacts(resolver, contactId);
836b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("          ", ".");
837b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
838b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
839b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
840b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
841b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    private static void dumpRawContacts(ContentResolver resolver, long contactId) {
842b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
843b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                RawContacts._ID,
844b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
845b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(RawContacts.CONTENT_URI, projection, RawContacts.CONTACT_ID + "="
846b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                + contactId, null, null);
847b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
848b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            long rawContactId = c.getLong(0);
849b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("RawContact", String.format("      %-5d", rawContactId));
850b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            dumpData(resolver, rawContactId);
851b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
852b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
853b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
854b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
855b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    private static void dumpData(ContentResolver resolver, long rawContactId) {
856b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
857b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.MIMETYPE,
858b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA1,
859b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA2,
860b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA3,
861b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
862b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(Data.CONTENT_URI, projection, Data.RAW_CONTACT_ID + "="
863b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                + rawContactId, null, Data.MIMETYPE);
864b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
865b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            String mimetype = c.getString(0);
866b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            if (Photo.CONTENT_ITEM_TYPE.equals(mimetype)) {
867b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Log.i("Photo     ", "");
868b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            } else {
869b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                mimetype = mimetype.substring(mimetype.indexOf('/') + 1);
870b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Log.i("Data      ", String.format("            %-10s %s,%s,%s", mimetype,
871b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                        c.getString(1), c.getString(2), c.getString(3)));
872b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            }
873b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
874b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
875b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
876285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov
87781d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov    protected void assertNetworkNotified(boolean expected) {
87881d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov        assertEquals(expected, ((SynchronousContactsProvider2)mActor.provider).isNetworkNotified());
87981d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov    }
8805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
8815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    /**
8825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * A contact in the database, and the attributes used to create it.  Construct using
8835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * {@link GoldenContactBuilder#build()}.
8845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     */
8855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    public final class GoldenContact {
8865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
8875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final long rawContactId;
8885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
8895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final long contactId;
8905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
8915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String givenName;
8925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
8935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String familyName;
8945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
8955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String nickname;
8965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
8975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final byte[] photo;
8985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
8995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String company;
9005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String title;
9025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String phone;
9045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String email;
9065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private GoldenContact(GoldenContactBuilder builder, long rawContactId, long contactId) {
9085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            this.rawContactId = rawContactId;
9105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            this.contactId = contactId;
9115b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            givenName = builder.givenName;
9125b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            familyName = builder.familyName;
9135b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            nickname = builder.nickname;
9145b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            photo = builder.photo;
9155b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            company = builder.company;
9165b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            title = builder.title;
9175b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            phone = builder.phone;
9185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            email = builder.email;
9195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public void delete() {
9225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            Uri rawContactUri = ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId);
9235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.delete(rawContactUri, null, null);
9245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the index of the contact in table "raw_contacts"
9285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public long getRawContactId() {
9305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return rawContactId;
9315b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9325b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9335b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9345b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the index of the contact in table "contacts"
9355b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9365b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public long getContactId() {
9375b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return contactId;
9385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9412d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill         * Returns the lookup key for the contact.
9422d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill         */
9432d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        public String getLookupKey() {
9442d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill            return queryLookupKey(contactId);
9452d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        }
9462d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill
9472d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        /**
9485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's given name.
9495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getGivenName() {
9515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return givenName;
9525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's family name.
9565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getFamilyName() {
9585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return familyName;
9595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's nickname.
9635b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getNickname() {
9655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return nickname;
9665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Return's the contact's photo
9705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public byte[] getPhoto() {
9725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return photo;
9735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Return's the company at which the contact works.
9775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getCompany() {
9795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return company;
9805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's job title.
9845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getTitle() {
9865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return title;
9875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's phone number
9915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getPhone() {
9935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return phone;
9945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
9955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
9965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
9975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's email address
9985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
9995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getEmail() {
10005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return email;
10015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     }
10035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    /**
10055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * Builds {@link GoldenContact} objects.  Unspecified boolean objects default to false.
10065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * Unspecified String objects default to null.
10075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     */
10085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    public final class GoldenContactBuilder {
10095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String givenName;
10115b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10125b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String familyName;
10135b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10145b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String nickname;
10155b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10165b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private byte[] photo;
10175b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String company;
10195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String title;
10215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String phone;
10235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String email;
10255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's given and family names.
10285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         *
10295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * TODO(dplotnikov): inline, or should we require them to set both names if they set either?
10305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10315b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder name(String givenName, String familyName) {
10325b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return givenName(givenName).familyName(familyName);
10335b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10345b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10355b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10365b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's given name.
10375b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder givenName(String value) {
10395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            givenName = value;
10405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
10415b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10435b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10445b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's family name.
10455b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10465b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder familyName(String value) {
10475b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            familyName = value;
10485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
10495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's nickname.
10535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder nickname(String value) {
10555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            nickname = value;
10565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
10575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's photo.
10615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder photo(byte[] value) {
10635b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            photo = value;
10645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
10655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The company at which the contact works.
10695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder company(String value) {
10715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            company = value;
10725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
10735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's job title.
10775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder title(String value) {
10795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            title = value;
10805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
10815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's phone number.
10855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder phone(String value) {
10875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            phone = value;
10885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
10895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's email address; also sets their IM status to {@link StatusUpdates#OFFLINE}
10935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * with a presence of "Coding for Android".
10945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder email(String value) {
10965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            email = value;
10975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
10985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Builds the {@link GoldenContact} specified by this builder.
11025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContact build() {
11045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            final long groupId = createGroup(mAccount, "gsid1", "title1");
11065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            long rawContactId = createRawContact();
11085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            insertGroupMembership(rawContactId, groupId);
11095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (givenName != null || familyName != null) {
11115b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertStructuredName(rawContactId, givenName, familyName);
11125b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
11135b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (nickname != null) {
11145b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertNickname(rawContactId, nickname);
11155b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
11165b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (photo != null) {
11175b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertPhoto(rawContactId);
11185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
11195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (company != null || title != null) {
11205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertOrganization(rawContactId);
11215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
11225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (email != null) {
11235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertEmail(rawContactId);
11245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
11255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (phone != null) {
11265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertPhone(rawContactId);
11275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
11285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            long contactId = queryContactId(rawContactId);
11305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11315b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return new GoldenContact(this, rawContactId, contactId);
11325b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11335b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11345b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertPhoto(long rawContactId) {
11355b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
11365b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
11375b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Photo.CONTENT_ITEM_TYPE);
11385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Photo.PHOTO, photo);
11395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
11405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11415b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertOrganization(long rawContactId) {
11435b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11445b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
11455b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
11465b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Organization.CONTENT_ITEM_TYPE);
11475b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Organization.TYPE, Organization.TYPE_WORK);
11485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (company != null) {
11495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                values.put(Organization.COMPANY, company);
11505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
11515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (title != null) {
11525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                values.put(Organization.TITLE, title);
11535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
11545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
11555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertEmail(long rawContactId) {
11585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
11605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
11615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Email.CONTENT_ITEM_TYPE);
11625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Email.TYPE, Email.TYPE_WORK);
11635b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Email.DATA, "foo@acme.com");
11645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
11655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            int protocol = Im.PROTOCOL_GOOGLE_TALK;
11675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.clear();
11695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PROTOCOL, protocol);
11705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.IM_HANDLE, email);
11715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.IM_ACCOUNT, "foo");
11725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PRESENCE_STATUS, StatusUpdates.OFFLINE);
11735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PRESENCE_CUSTOM_STATUS, "Coding for Android");
11745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(StatusUpdates.CONTENT_URI, values);
11755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertPhone(long rawContactId) {
11785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
11795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
11805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE);
11815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.IS_PRIMARY, 1);
11825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Phone.TYPE, Phone.TYPE_HOME);
11835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Phone.NUMBER, phone);
11845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
11855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    }
1187d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov}
1188