BaseContactsProvider2Test.java revision 92ddc5cdc4d89ee2c6e861ae7b3a3a913ffa0100
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
2109ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport com.google.android.collect.Sets;
2209ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov
23e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.accounts.Account;
24bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikovimport android.content.ContentProvider;
25b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikovimport android.content.ContentResolver;
26d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.content.ContentUris;
27d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.content.ContentValues;
28e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.content.Context;
299261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport android.content.Entity;
30a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport android.content.res.Resources;
31d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.database.Cursor;
32d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.net.Uri;
33bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikovimport android.provider.ContactsContract;
34d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.provider.ContactsContract.AggregationExceptions;
35a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Email;
366d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Event;
379261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport android.provider.ContactsContract.CommonDataKinds.GroupMembership;
38e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Im;
39a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Nickname;
406d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Note;
4101911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Organization;
42bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.Phone;
43ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millarimport android.provider.ContactsContract.CommonDataKinds.Photo;
44e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.StructuredName;
454a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikovimport android.provider.ContactsContract.CommonDataKinds.StructuredPostal;
4609ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.provider.ContactsContract.Contacts;
4709ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.provider.ContactsContract.Data;
4809ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.provider.ContactsContract.Groups;
4909ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.provider.ContactsContract.RawContacts;
5009ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.provider.ContactsContract.Settings;
5109ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.provider.ContactsContract.StatusUpdates;
52d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.test.AndroidTestCase;
5309ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.test.MoreAsserts;
54d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.test.mock.MockContentResolver;
55b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikovimport android.util.Log;
56bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
57a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport java.io.ByteArrayOutputStream;
58a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport java.io.IOException;
59a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikovimport java.io.InputStream;
609261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport java.util.ArrayList;
619261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport java.util.Arrays;
62e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport java.util.Comparator;
63ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikovimport java.util.Iterator;
6420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikovimport java.util.Map;
65ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikovimport java.util.Map.Entry;
6609ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport java.util.Set;
67d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
68d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov/**
69d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * A common superclass for {@link ContactsProvider2}-related tests.
70d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov */
71d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovpublic abstract class BaseContactsProvider2Test extends AndroidTestCase {
72d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
73d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected static final String PACKAGE = "ContactsProvider2Test";
7473f1f396c155b247b903d8f4111db17d3e13dc4dDmitri Plotnikov    public static final String READ_ONLY_ACCOUNT_TYPE =
7573f1f396c155b247b903d8f4111db17d3e13dc4dDmitri Plotnikov            SynchronousContactsProvider2.READ_ONLY_ACCOUNT_TYPE;
76d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
77bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected ContactsActor mActor;
78d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected MockContentResolver mResolver;
799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected Account mAccount = new Account("account1", "account type1");
80e2579e029472f76b2dfda141444d775c67da0ec8Cynthia Wong    protected Account mAccountTwo = new Account("account2", "account type2");
819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
822482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov    private byte[] mTestPhoto;
832482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov
849261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static Long NO_LONG = new Long(0);
859261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static String NO_STRING = new String("");
869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static Account NO_ACCOUNT = new Account("a", "b");
87d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
88bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected Class<? extends ContentProvider> getProviderClass() {
89bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        return SynchronousContactsProvider2.class;
90bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    }
91bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
92bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected String getAuthority() {
93bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        return ContactsContract.AUTHORITY;
94bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    }
95bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
96d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    @Override
97d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected void setUp() throws Exception {
98d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        super.setUp();
99d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
100bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        mActor = new ContactsActor(getContext(), PACKAGE_GREY, getProviderClass(), getAuthority());
101d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        mResolver = mActor.resolver;
102445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov        if (mActor.provider instanceof SynchronousContactsProvider2) {
1039a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov            getContactsProvider().wipeData();
104445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov        }
1059261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
1069261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
1070265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    @Override
1080265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    protected void tearDown() throws Exception {
1090265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov        if (mActor.provider instanceof SynchronousContactsProvider2) {
1109a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov            getContactsProvider().getDatabaseHelper(mActor.context).close();
1110265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov        }
1120265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov        super.tearDown();
1130265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    }
1140265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov
1159a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov    public SynchronousContactsProvider2 getContactsProvider() {
1169a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov        return (SynchronousContactsProvider2) mActor.provider;
1179a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov    }
1189a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov
119e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public Context getMockContext() {
120e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        return mActor.context;
121e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
122e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
123e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public void addAuthority(String authority) {
124e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        mActor.addAuthority(authority);
125e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
126e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
127e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public ContentProvider addProvider(Class<? extends ContentProvider> providerClass,
128e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov            String authority) throws Exception {
129e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        return mActor.addProvider(providerClass, authority);
130e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
131e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
1323d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov    public ContentProvider getProvider() {
1333d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov        return mActor.provider;
1343d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov    }
1353d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov
1369261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected Uri maybeAddAccountQueryParameters(Uri uri, Account account) {
1379261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == null) {
1389261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return uri;
1399261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
1409261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return uri.buildUpon()
141df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana                .appendQueryParameter(RawContacts.ACCOUNT_NAME, account.name)
142df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana                .appendQueryParameter(RawContacts.ACCOUNT_TYPE, account.type)
1439261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                .build();
144d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
145d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
146d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long createRawContact() {
147226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        return createRawContact(null);
148d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
149d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
150c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov    protected long createRawContactWithName() {
151e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName(null);
152e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
153e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
154e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected long createRawContactWithName(Account account) {
155e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName("John", "Doe", account);
1563cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov    }
1573cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov
1583cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov    protected long createRawContactWithName(String firstName, String lastName) {
159e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName(firstName, lastName, null);
160e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
161e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
162e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected long createRawContactWithName(String firstName, String lastName, Account account) {
163e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        long rawContactId = createRawContact(account);
1643cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        insertStructuredName(rawContactId, firstName, lastName);
165c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        return rawContactId;
166c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov    }
167c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov
168e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected Uri setCallerIsSyncAdapter(Uri uri, Account account) {
169e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        if (account == null) {
170e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey            return uri;
171e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        }
172e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        final Uri.Builder builder = uri.buildUpon();
173e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.RawContacts.ACCOUNT_NAME, account.name);
174e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.RawContacts.ACCOUNT_TYPE, account.type);
175e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true");
176e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return builder.build();
177e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
178e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
179226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana    protected long createRawContact(Account account, String... extras) {
1809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
181dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        extrasVarArgsToValues(values, extras);
182dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        final Uri uri = maybeAddAccountQueryParameters(RawContacts.CONTENT_URI, account);
183dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        Uri contactUri = mResolver.insert(uri, values);
184dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return ContentUris.parseId(contactUri);
185dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
186dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
187dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected int updateItem(Uri uri, long id, String... extras) {
188dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        Uri itemUri = ContentUris.withAppendedId(uri, id);
189dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return updateItem(itemUri, extras);
190dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
191dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
192dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected int updateItem(Uri uri, String... extras) {
193dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        ContentValues values = new ContentValues();
194dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        extrasVarArgsToValues(values, extras);
195dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return mResolver.update(uri, values, null, null);
196dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
197dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
198dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    private static void extrasVarArgsToValues(ContentValues values, String... extras) {
199226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        for (int i = 0; i < extras.length; ) {
200226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana            values.put(extras[i], extras[i + 1]);
201226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana            i += 2;
202226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        }
2039261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
2049261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
2059261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long createGroup(Account account, String sourceId, String title) {
206dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return createGroup(account, sourceId, title, 1, false, false);
2070be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov    }
2080be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov
2090be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov    protected long createGroup(Account account, String sourceId, String title, int visible) {
210dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return createGroup(account, sourceId, title, visible, false, false);
211dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
212dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
213f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov    protected long createAutoAddGroup(Account account) {
214f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov        return createGroup(account, "auto", "auto",
215f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov                0 /* visible */,  true /* auto-add */, false /* fav */);
216f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov    }
217f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov
218dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected long createGroup(Account account, String sourceId, String title,
219dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            int visible, boolean autoAdd, boolean favorite) {
2209261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
2219261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Groups.SOURCE_ID, sourceId);
2229261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Groups.TITLE, title);
2230be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov        values.put(Groups.GROUP_VISIBLE, visible);
224dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        values.put(Groups.AUTO_ADD, autoAdd ? 1 : 0);
225dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        values.put(Groups.FAVORITES, favorite ? 1 : 0);
2269261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        final Uri uri = maybeAddAccountQueryParameters(Groups.CONTENT_URI, account);
2279261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return ContentUris.parseId(mResolver.insert(uri, values));
2289261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
2299261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
23089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected void createSettings(Account account, String shouldSync, String ungroupedVisible) {
23189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        ContentValues values = new ContentValues();
23289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.ACCOUNT_NAME, account.name);
23389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.ACCOUNT_TYPE, account.type);
23489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.SHOULD_SYNC, shouldSync);
23589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.UNGROUPED_VISIBLE, ungroupedVisible);
23689c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        mResolver.insert(Settings.CONTENT_URI, values);
23789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
23889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
2395ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertStructuredName(long rawContactId, String givenName, String familyName) {
240d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        ContentValues values = new ContentValues();
241d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        StringBuilder sb = new StringBuilder();
242d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (givenName != null) {
243d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(givenName);
244d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
245d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (givenName != null && familyName != null) {
246d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(" ");
247d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
248d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (familyName != null) {
249d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(familyName);
250d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
251d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.DISPLAY_NAME, sb.toString());
252d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.GIVEN_NAME, givenName);
253d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.FAMILY_NAME, familyName);
254d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
2555ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        return insertStructuredName(rawContactId, values);
2564097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov    }
2574097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
2585ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertStructuredName(long rawContactId, ContentValues values) {
2595ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
2604097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        values.put(Data.MIMETYPE, StructuredName.CONTENT_ITEM_TYPE);
261d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
262d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        return resultUri;
263d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
264d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
265a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka    protected Uri insertOrganization(long rawContactId, ContentValues values) {
266a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka        return insertOrganization(rawContactId, values, false);
26701911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    }
26801911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
269a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka    protected Uri insertOrganization(long rawContactId, ContentValues values, boolean primary) {
27001911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
27101911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Data.MIMETYPE, Organization.CONTENT_ITEM_TYPE);
27201911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Organization.TYPE, Organization.TYPE_WORK);
27301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        if (primary) {
2740265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
27501911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        }
27601911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
27701911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
27801911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        return resultUri;
27901911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    }
28001911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
2815ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber) {
28225abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        return insertPhoneNumber(rawContactId, phoneNumber, false);
28325abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    }
28425abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov
28525abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber, boolean primary) {
286bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        ContentValues values = new ContentValues();
2875ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
288bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        values.put(Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE);
289bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        values.put(Phone.NUMBER, phoneNumber);
2903cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Phone.TYPE, Phone.TYPE_HOME);
29125abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        if (primary) {
2920265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
29325abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        }
294bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
295bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
296bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        return resultUri;
297bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov    }
298bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
2995ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email) {
30025abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        return insertEmail(rawContactId, email, false);
30125abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    }
30225abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov
30325abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email, boolean primary) {
304916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        return insertEmail(rawContactId, email, primary, Email.TYPE_HOME, null);
305916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov    }
306916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov
307916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email, boolean primary, int type,
308916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov            String label) {
309a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        ContentValues values = new ContentValues();
3105ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
311a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Data.MIMETYPE, Email.CONTENT_ITEM_TYPE);
312a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Email.DATA, email);
313916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        values.put(Email.TYPE, type);
314916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        values.put(Email.LABEL, label);
31525abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        if (primary) {
316916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
31725abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        }
318a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
319a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
320a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        return resultUri;
321a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov    }
322a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
3235ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertNickname(long rawContactId, String nickname) {
324a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        ContentValues values = new ContentValues();
3255ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
326a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Data.MIMETYPE, Nickname.CONTENT_ITEM_TYPE);
327a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Nickname.NAME, nickname);
3283cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Nickname.TYPE, Nickname.TYPE_OTHER_NAME);
329a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
330a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
331a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        return resultUri;
332a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov    }
333a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
3344a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov    protected Uri insertPostalAddress(long rawContactId, String formattedAddress) {
3354a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        ContentValues values = new ContentValues();
3364a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3374a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(Data.MIMETYPE, StructuredPostal.CONTENT_ITEM_TYPE);
3384a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(StructuredPostal.FORMATTED_ADDRESS, formattedAddress);
3394a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov
3404a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
3414a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        return resultUri;
3424a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov    }
3434a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov
344eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov    protected Uri insertPostalAddress(long rawContactId, ContentValues values) {
345eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
346eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov        values.put(Data.MIMETYPE, StructuredPostal.CONTENT_ITEM_TYPE);
347eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
348eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov        return resultUri;
349eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov    }
350eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov
3515ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertPhoto(long rawContactId) {
352ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        ContentValues values = new ContentValues();
3535ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
354ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        values.put(Data.MIMETYPE, Photo.CONTENT_ITEM_TYPE);
3552482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        values.put(Photo.PHOTO, loadTestPhoto());
356ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
357ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        return resultUri;
358ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
359ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
3605ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertGroupMembership(long rawContactId, String sourceId) {
3619261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
3625ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3639261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Data.MIMETYPE, GroupMembership.CONTENT_ITEM_TYPE);
3649261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(GroupMembership.GROUP_SOURCE_ID, sourceId);
3659261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return mResolver.insert(Data.CONTENT_URI, values);
3669261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
3679261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
3685ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertGroupMembership(long rawContactId, Long groupId) {
3699261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
3705ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3719261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Data.MIMETYPE, GroupMembership.CONTENT_ITEM_TYPE);
3729261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(GroupMembership.GROUP_ROW_ID, groupId);
3739261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return mResolver.insert(Data.CONTENT_URI, values);
3749261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
3759261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
376f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov    public void removeGroupMemberships(long rawContactId) {
377f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov        mResolver.delete(Data.CONTENT_URI,
378f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov                Data.MIMETYPE + "=? AND " + GroupMembership.RAW_CONTACT_ID + "=?",
379f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov                new String[] { GroupMembership.CONTENT_ITEM_TYPE, String.valueOf(rawContactId) });
380f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov    }
381f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov
38282bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov    protected Uri insertStatusUpdate(int protocol, String customProtocol, String handle,
383aabcd1d34a71ad06ee0a9395331540484f1ceb17Vasu Nori            int presence, String status, int chatMode) {
384aabcd1d34a71ad06ee0a9395331540484f1ceb17Vasu Nori        return insertStatusUpdate(protocol, customProtocol, handle, presence, status, 0, chatMode);
385a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov    }
386a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov
38782bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov    protected Uri insertStatusUpdate(int protocol, String customProtocol, String handle,
388aabcd1d34a71ad06ee0a9395331540484f1ceb17Vasu Nori            int presence, String status, long timestamp, int chatMode) {
389bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        ContentValues values = new ContentValues();
39082bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.PROTOCOL, protocol);
39182bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.CUSTOM_PROTOCOL, customProtocol);
39282bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.IM_HANDLE, handle);
393f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov        return insertStatusUpdate(values, presence, status, timestamp, chatMode);
394f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    }
395f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov
396f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    protected Uri insertStatusUpdate(
397f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov            long dataId, int presence, String status, long timestamp, int chatMode) {
398f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov        ContentValues values = new ContentValues();
399f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov        values.put(StatusUpdates.DATA_ID, dataId);
400f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov        return insertStatusUpdate(values, presence, status, timestamp, chatMode);
401f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    }
402f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov
403f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    private Uri insertStatusUpdate(
404f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov            ContentValues values, int presence, String status, long timestamp, int chatMode) {
405a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (presence != 0) {
40682bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.PRESENCE, presence);
407aabcd1d34a71ad06ee0a9395331540484f1ceb17Vasu Nori            values.put(StatusUpdates.CHAT_CAPABILITY, chatMode);
408a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
409a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (status != null) {
41082bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.STATUS, status);
411a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
412a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (timestamp != 0) {
41382bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.STATUS_TIMESTAMP, timestamp);
414a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
415bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
41682bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        Uri resultUri = mResolver.insert(StatusUpdates.CONTENT_URI, values);
417bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        return resultUri;
418bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
419bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
4204dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov    protected Uri insertImHandle(long rawContactId, int protocol, String customProtocol,
4214dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov            String handle) {
422bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        ContentValues values = new ContentValues();
4235ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
424bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Data.MIMETYPE, Im.CONTENT_ITEM_TYPE);
425bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Im.PROTOCOL, protocol);
4264dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov        values.put(Im.CUSTOM_PROTOCOL, customProtocol);
427bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Im.DATA, handle);
4283cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Im.TYPE, Im.TYPE_HOME);
429bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
430bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
431bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        return resultUri;
432bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
433bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
4346d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov    protected Uri insertEvent(long rawContactId, int type, String date) {
4356d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        ContentValues values = new ContentValues();
4366d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
4376d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Data.MIMETYPE, Event.CONTENT_ITEM_TYPE);
4386d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Event.TYPE, type);
4396d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Event.START_DATE, date);
4406d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
4416d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        return resultUri;
4426d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov    }
4436d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov
4446d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov    protected Uri insertNote(long rawContactId, String note) {
4456d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        ContentValues values = new ContentValues();
4466d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
4476d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Data.MIMETYPE, Note.CONTENT_ITEM_TYPE);
4486d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Note.NOTE, note);
4496d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
4506d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        return resultUri;
4516d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov    }
4526d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov
453767e109d6f08749b9ed202c0b71f3459eaae2115Dmitri Plotnikov    protected void setContactAccount(long rawContactId, String accountType, String accountName) {
454ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        ContentValues values = new ContentValues();
455767e109d6f08749b9ed202c0b71f3459eaae2115Dmitri Plotnikov        values.put(RawContacts.ACCOUNT_TYPE, accountType);
4566cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov        values.put(RawContacts.ACCOUNT_NAME, accountName);
457ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
458ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        mResolver.update(ContentUris.withAppendedId(
4595ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov                RawContacts.CONTENT_URI, rawContactId), values, null, null);
460ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
461ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
4620c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov    protected void setAggregationException(int type, long rawContactId1, long rawContactId2) {
463d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        ContentValues values = new ContentValues();
4640c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        values.put(AggregationExceptions.RAW_CONTACT_ID1, rawContactId1);
4650c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        values.put(AggregationExceptions.RAW_CONTACT_ID2, rawContactId2);
466d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(AggregationExceptions.TYPE, type);
4673d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov        assertEquals(1, mResolver.update(AggregationExceptions.CONTENT_URI, values, null, null));
468d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
469d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
470d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected Cursor queryRawContact(long rawContactId) {
4710c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
4720c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov                null, null, null, null);
473d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
474d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
475d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected Cursor queryContact(long contactId) {
476d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
477d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov                null, null, null, null);
478d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
479d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
480af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov    protected Cursor queryContact(long contactId, String[] projection) {
481af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
482af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov                projection, null, null, null);
483bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
484bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
485f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    protected Uri getContactUriForRawContact(long rawContactId) {
486f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov        return ContentUris.withAppendedId(Contacts.CONTENT_URI, queryContactId(rawContactId));
487f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    }
488f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov
489d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long queryContactId(long rawContactId) {
490d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryRawContact(rawContactId);
491d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertTrue(c.moveToFirst());
492d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId = c.getLong(c.getColumnIndex(RawContacts.CONTACT_ID));
493d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        c.close();
494d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        return contactId;
495d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
496d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
497d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long queryPhotoId(long contactId) {
498d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryContact(contactId);
499ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        assertTrue(c.moveToFirst());
500d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long photoId = c.getInt(c.getColumnIndex(Contacts.PHOTO_ID));
501ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        c.close();
502ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        return photoId;
503ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
504ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
505dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected boolean queryRawContactIsStarred(long rawContactId) {
506dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        Cursor c = queryRawContact(rawContactId);
507dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        try {
508dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            assertTrue(c.moveToFirst());
509dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            return c.getLong(c.getColumnIndex(RawContacts.STARRED)) != 0;
510dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        } finally {
511dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            c.close();
512dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        }
513dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
514dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
515d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected String queryDisplayName(long contactId) {
516d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryContact(contactId);
517d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertTrue(c.moveToFirst());
518d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        String displayName = c.getString(c.getColumnIndex(Contacts.DISPLAY_NAME));
519d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        c.close();
520d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        return displayName;
521d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
522d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
5232149ab82f021c204618d0d3644e261fd7a8d8490Dmitri Plotnikov    protected String queryLookupKey(long contactId) {
5242d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        Cursor c = queryContact(contactId);
5252d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        assertTrue(c.moveToFirst());
5262d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        String lookupKey = c.getString(c.getColumnIndex(Contacts.LOOKUP_KEY));
5272d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        c.close();
5282d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        return lookupKey;
5292d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill    }
5302d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill
531d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertAggregated(long rawContactId1, long rawContactId2) {
532d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
533d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
534d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 == contactId2);
535bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov    }
536bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
537d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertAggregated(long rawContactId1, long rawContactId2,
538d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov            String expectedDisplayName) {
539d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
540d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
541d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 == contactId2);
542d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
543d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        String displayName = queryDisplayName(contactId1);
544d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertEquals(expectedDisplayName, displayName);
545d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
546d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
547d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertNotAggregated(long rawContactId1, long rawContactId2) {
548d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
549d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
550d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 != contactId2);
551d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
5524097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
5535ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected void assertStructuredName(long rawContactId, String prefix, String givenName,
5544097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov            String middleName, String familyName, String suffix) {
555f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov        Uri uri = Uri.withAppendedPath(
556f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov                ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
5576cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov                RawContacts.Data.CONTENT_DIRECTORY);
5584097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
5594097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        final String[] projection = new String[] {
5604097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                StructuredName.PREFIX, StructuredName.GIVEN_NAME, StructuredName.MIDDLE_NAME,
5614097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                StructuredName.FAMILY_NAME, StructuredName.SUFFIX
5624097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        };
5634097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
5644097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        Cursor c = mResolver.query(uri, projection, Data.MIMETYPE + "='"
5654097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                + StructuredName.CONTENT_ITEM_TYPE + "'", null, null);
5664097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
5674097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertTrue(c.moveToFirst());
5684097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(prefix, c.getString(0));
5694097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(givenName, c.getString(1));
5704097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(middleName, c.getString(2));
5714097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(familyName, c.getString(3));
5724097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(suffix, c.getString(4));
5734097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        c.close();
5744097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov    }
5759261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5769261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long assertSingleGroup(Long rowId, Account account, String sourceId, String title) {
5779261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(Groups.CONTENT_URI, null, null, null, null);
5789261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        try {
5799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.moveToNext());
5809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long actualRowId = assertGroup(c, rowId, account, sourceId, title);
5819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertFalse(c.moveToNext());
5829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return actualRowId;
5839261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } finally {
5849261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            c.close();
5859261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
5879261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
5885ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected long assertSingleGroupMembership(Long rowId, Long rawContactId, Long groupRowId,
5899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String sourceId) {
5909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(ContactsContract.Data.CONTENT_URI, null, null, null, null);
5919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        try {
5929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.moveToNext());
5935ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov            long actualRowId = assertGroupMembership(c, rowId, rawContactId, groupRowId, sourceId);
5949261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertFalse(c.moveToNext());
5959261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return actualRowId;
5969261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } finally {
5979261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            c.close();
5989261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
5999261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6015ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected long assertGroupMembership(Cursor c, Long rowId, Long rawContactId, Long groupRowId,
6029261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String sourceId) {
6039261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, rowId, Data._ID);
6045ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        assertNullOrEquals(c, rawContactId, GroupMembership.RAW_CONTACT_ID);
6059261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, groupRowId, GroupMembership.GROUP_ROW_ID);
6069261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, sourceId, GroupMembership.GROUP_SOURCE_ID);
6079261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return c.getLong(c.getColumnIndexOrThrow("_id"));
6089261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6099261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6109261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long assertGroup(Cursor c, Long rowId, Account account, String sourceId, String title) {
6119261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, rowId, Groups._ID);
6129261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, account);
6139261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, sourceId, Groups.SOURCE_ID);
6149261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, title, Groups.TITLE);
6159261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return c.getLong(c.getColumnIndexOrThrow("_id"));
6169261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6179261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6189261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, Account account) {
6199261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == NO_ACCOUNT) {
6209261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return;
6219261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6229261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == null) {
6239261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.isNull(c.getColumnIndexOrThrow(Groups.ACCOUNT_NAME)));
6249261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.isNull(c.getColumnIndexOrThrow(Groups.ACCOUNT_TYPE)));
6259261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } else {
626df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana            assertEquals(account.name, c.getString(c.getColumnIndexOrThrow(Groups.ACCOUNT_NAME)));
627df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana            assertEquals(account.type, c.getString(c.getColumnIndexOrThrow(Groups.ACCOUNT_TYPE)));
6289261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6299261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6309261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6319261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, Long value, String columnName) {
6329261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (value != NO_LONG) {
6339261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (value == null) assertTrue(c.isNull(c.getColumnIndexOrThrow(columnName)));
6349261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            else assertEquals((long) value, c.getLong(c.getColumnIndexOrThrow(columnName)));
6359261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6369261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6379261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6389261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, String value, String columnName) {
6399261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (value != NO_STRING) {
6409261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (value == null) assertTrue(c.isNull(c.getColumnIndexOrThrow(columnName)));
6419261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            else assertEquals(value, c.getString(c.getColumnIndexOrThrow(columnName)));
6429261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6439261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6449261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6459261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void assertDataRow(ContentValues actual, String expectedMimetype,
6469261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            Object... expectedArguments) {
6479261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertEquals(actual.toString(), expectedMimetype, actual.getAsString(Data.MIMETYPE));
6489261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        for (int i = 0; i < expectedArguments.length; i += 2) {
6499261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String columnName = (String) expectedArguments[i];
6509261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            Object expectedValue = expectedArguments[i + 1];
6519261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue instanceof Uri) {
6529261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                expectedValue = ContentUris.parseId((Uri) expectedValue);
6539261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
6549261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue == null) {
6559261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                assertNull(actual.toString(), actual.get(columnName));
6569261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
6579261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue instanceof Long) {
658226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
659226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsLong(columnName));
6609261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else if (expectedValue instanceof Integer) {
661226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
662226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsInteger(columnName));
6639261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else if (expectedValue instanceof String) {
664226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
665226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsString(columnName));
6669261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else {
667226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
668226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.get(columnName));
6699261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
6709261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6719261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6729261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
673dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected void assertNoRowsAndClose(Cursor c) {
674dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        try {
675dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            assertFalse(c.moveToNext());
676dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        } finally {
677dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            c.close();
678dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        }
679dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
680dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
6819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected static class IdComparator implements Comparator<ContentValues> {
6829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        public int compare(ContentValues o1, ContentValues o2) {
6839261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long id1 = o1.getAsLong(ContactsContract.Data._ID);
6849261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long id2 = o2.getAsLong(ContactsContract.Data._ID);
6859261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (id1 == id2) return 0;
6869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return (id1 < id2) ? -1 : 1;
6879261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6889261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected ContentValues[] asSortedContentValuesArray(
6919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            ArrayList<Entity.NamedContentValues> subValues) {
6929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues[] result = new ContentValues[subValues.size()];
6939261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        int i = 0;
6949261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        for (Entity.NamedContentValues subValue : subValues) {
6959261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            result[i] = subValue.values;
6969261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            i++;
6979261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6989261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Arrays.sort(result, new IdComparator());
6999261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return result;
7009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7019261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7029261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void assertDirty(Uri uri, boolean state) {
7039261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(uri, new String[]{"dirty"}, null, null, null);
7049261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertTrue(c.moveToNext());
7059261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertEquals(state, c.getLong(0) != 0);
7069261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertFalse(c.moveToNext());
7073cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        c.close();
7089261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7099261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7109261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long getVersion(Uri uri) {
7119261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(uri, new String[]{"version"}, null, null, null);
7129261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertTrue(c.moveToNext());
7139261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        long version = c.getLong(0);
7149261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertFalse(c.moveToNext());
7153cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        c.close();
7169261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return version;
7179261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7189261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7199261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void clearDirty(Uri uri) {
7209261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
7219261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put("dirty", 0);
7229261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        mResolver.update(uri, values, null, null);
7239261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
72420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
7255870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    protected void storeValue(Uri contentUri, long id, String column, String value) {
726f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        storeValue(ContentUris.withAppendedId(contentUri, id), column, value);
727f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
728f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
729f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, String column, String value) {
730f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        ContentValues values = new ContentValues();
731f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        values.put(column, value);
732f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
733f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        mResolver.update(contentUri, values, null, null);
734f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
735f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
736f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, long id, String column, long value) {
737f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        storeValue(ContentUris.withAppendedId(contentUri, id), column, value);
738f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
739f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
740f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, String column, long value) {
7415870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        ContentValues values = new ContentValues();
7425870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        values.put(column, value);
7435870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov
744f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        mResolver.update(contentUri, values, null, null);
7455870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    }
7465870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov
747f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov    protected void assertStoredValue(Uri contentUri, long id, String column, Object expectedValue) {
748f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov        assertStoredValue(ContentUris.withAppendedId(contentUri, id), column, expectedValue);
749f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov    }
750f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov
7515870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    protected void assertStoredValue(Uri rowUri, String column, Object expectedValue) {
75220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        String value = getStoredValue(rowUri, column);
75301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        if (expectedValue == null) {
75401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            assertNull("Column value " + column, value);
75501911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        } else {
75601911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            assertEquals("Column value " + column, String.valueOf(expectedValue), value);
75701911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        }
75820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
75920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
76089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected void assertStoredValue(Uri rowUri, String selection, String[] selectionArgs,
76189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            String column, Object expectedValue) {
76289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        String value = getStoredValue(rowUri, selection, selectionArgs, column);
76389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        if (expectedValue == null) {
76489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            assertNull("Column value " + column, value);
76589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        } else {
76689c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            assertEquals("Column value " + column, String.valueOf(expectedValue), value);
76789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        }
76889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
76989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
77020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected String getStoredValue(Uri rowUri, String column) {
77189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        return getStoredValue(rowUri, null, null, column);
77289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
77389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
77489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected String getStoredValue(Uri uri, String selection, String[] selectionArgs,
77589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            String column) {
7768e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov        String value = null;
77789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        Cursor c = mResolver.query(uri, new String[] { column }, selection, selectionArgs, null);
77820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
77992ddc5cdc4d89ee2c6e861ae7b3a3a913ffa0100Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
78092ddc5cdc4d89ee2c6e861ae7b3a3a913ffa0100Dmitri Plotnikov
7818e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov            if (c.moveToFirst()) {
7828e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov                value = c.getString(c.getColumnIndex(column));
7838e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov            }
78420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
78520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
78620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
78720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        return value;
78820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
78920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
79020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertStoredValues(Uri rowUri, ContentValues expectedValues) {
791bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov        assertStoredValues(rowUri, null, null, expectedValues);
792bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov    }
793bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov
794892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    protected void assertStoredValues(Uri rowUri, ContentValues[] expectedValues) {
795892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        assertStoredValues(rowUri, null, null, expectedValues);
796892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    }
797892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov
798bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov    protected void assertStoredValues(Uri rowUri, String selection, String[] selectionArgs,
799bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov            ContentValues expectedValues) {
800bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov        Cursor c = mResolver.query(rowUri, null, selection, selectionArgs, null);
80120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
80220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
80320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.moveToFirst();
80420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertCursorValues(c, expectedValues);
80520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
80620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
80720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
80820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
80920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
810ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    protected void assertStoredValuesWithProjection(Uri rowUri, ContentValues expectedValues) {
811ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Cursor c = mResolver.query(rowUri, buildProjection(expectedValues), null, null, null);
812ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        try {
813ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
814ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            c.moveToFirst();
815ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            assertCursorValues(c, expectedValues);
816ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        } finally {
817ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            c.close();
818ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        }
819ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
820ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
821892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    protected void assertStoredValues(
822892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            Uri rowUri, String selection, String[] selectionArgs, ContentValues[] expectedValues) {
823892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        Cursor c = mResolver.query(rowUri, null, selection, selectionArgs, null);
824892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        try {
825892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            assertEquals("Record count", expectedValues.length, c.getCount());
826892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            assertCursorValues(c, expectedValues);
827892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        } finally {
828892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            c.close();
829892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        }
830892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    }
831892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov
83220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    /**
83320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * Constructs a selection (where clause) out of all supplied values, uses it
83420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * to query the provider and verifies that a single row is returned and it
83520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * has the same values as requested.
83620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     */
83720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertSelection(Uri uri, ContentValues values, String idColumn, long id) {
838ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        assertSelection(uri, values, idColumn, id, null);
839ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
840ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
841ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    public void assertSelectionWithProjection(Uri uri, ContentValues values, String idColumn,
842ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            long id) {
843ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        assertSelection(uri, values, idColumn, id, buildProjection(values));
844ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
845ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
846ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    private void assertSelection(Uri uri, ContentValues values, String idColumn, long id,
847ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            String[] projection) {
84820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        StringBuilder sb = new StringBuilder();
84920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        ArrayList<String> selectionArgs = new ArrayList<String>(values.size());
850c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        if (idColumn != null) {
851c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            sb.append(idColumn).append("=").append(id);
852c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        }
85320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        Set<Map.Entry<String, Object>> entries = values.valueSet();
85420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        for (Map.Entry<String, Object> entry : entries) {
85520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String column = entry.getKey();
85620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            Object value = entry.getValue();
857c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            if (sb.length() != 0) {
858c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov                sb.append(" AND ");
859c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            }
860c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            sb.append(column);
86120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            if (value == null) {
86220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                sb.append(" IS NULL");
86320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            } else {
86420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                sb.append("=?");
86520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                selectionArgs.add(String.valueOf(value));
86620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            }
86720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
86820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
869ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Cursor c = mResolver.query(uri, projection, sb.toString(), selectionArgs.toArray(new String[0]),
87020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                null);
87120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
87220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
87320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.moveToFirst();
87420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertCursorValues(c, values);
87520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
87620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
87720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
87820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
87920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
880c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov    protected void assertCursorValue(Cursor cursor, String column, Object expectedValue) {
881c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov        String actualValue = cursor.getString(cursor.getColumnIndex(column));
882c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov        assertEquals("Column " + column, String.valueOf(expectedValue),
883c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov                String.valueOf(actualValue));
884c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov    }
885c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov
88620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertCursorValues(Cursor cursor, ContentValues expectedValues) {
887892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        StringBuilder message = new StringBuilder();
888892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        boolean result = equalsWithExpectedValues(cursor, expectedValues, message);
889892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        assertTrue(message.toString(), result);
890892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    }
891892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov
892892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    protected void assertCursorValues(Cursor cursor, ContentValues[] expectedValues) {
893892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        StringBuilder message = new StringBuilder();
894892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        for (ContentValues v : expectedValues) {
895892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            boolean found = false;
896892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            cursor.moveToPosition(-1);
897892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            while (cursor.moveToNext()) {
898892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                found = equalsWithExpectedValues(cursor, v, message);
899892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                if (found) {
900892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                    break;
901892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                }
902892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            }
903892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            assertTrue("Expected values can not be found " + v + message.toString(), found);
904892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        }
905892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    }
906892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov
907892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    private boolean equalsWithExpectedValues(Cursor cursor, ContentValues expectedValues,
908892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            StringBuilder msgBuffer) {
90920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        Set<Map.Entry<String, Object>> entries = expectedValues.valueSet();
91020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        for (Map.Entry<String, Object> entry : entries) {
91120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String column = entry.getKey();
91220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            int index = cursor.getColumnIndex(column);
913892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            if (index == -1) {
914892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                msgBuffer.append("No such column: ").append(column);
915892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                return false;
916892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            }
91720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            Object expectedValue = expectedValues.get(column);
91820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String value;
91920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            if (expectedValue instanceof byte[]) {
92020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                expectedValue = Hex.encodeHex((byte[])expectedValue, false);
92120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                value = Hex.encodeHex(cursor.getBlob(index), false);
92220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            } else {
92320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                expectedValue = expectedValues.getAsString(column);
92420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                value = cursor.getString(index);
92520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            }
926892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            if (expectedValue != null && !expectedValue.equals(value) || value != null
927892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                    && !value.equals(expectedValue)) {
928892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                msgBuffer
929892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append("Column value ")
930892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(column)
931892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(" expected <")
932892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(expectedValue)
933892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(">, but was <")
934892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(value)
935892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append('>');
936892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                return false;
937892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            }
93820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
939892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        return true;
94020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
94133b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov
942ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    private String[] buildProjection(ContentValues values) {
943ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        String[] projection = new String[values.size()];
944ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Iterator<Entry<String, Object>> iter = values.valueSet().iterator();
945ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        for (int i = 0; i < projection.length; i++) {
946ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            projection[i] = iter.next().getKey();
947ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        }
948ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        return projection;
949ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
950ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
95133b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov    protected int getCount(Uri uri, String selection, String[] selectionArgs) {
95233b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        Cursor c = mResolver.query(uri, null, selection, selectionArgs, null);
95333b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        try {
95433b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov            return c.getCount();
95533b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        } finally {
95633b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov            c.close();
95733b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        }
95833b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov    }
959a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov
9602482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov    protected byte[] loadTestPhoto() {
9612482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        if (mTestPhoto == null) {
9622482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            final Resources resources = getContext().getResources();
9632482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            InputStream is = resources
9642482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                    .openRawResource(com.android.internal.R.drawable.ic_contact_picture);
9652482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            ByteArrayOutputStream os = new ByteArrayOutputStream();
9662482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            byte[] buffer = new byte[1000];
9672482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            int count;
9682482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            try {
9692482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                while ((count = is.read(buffer)) != -1) {
9702482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                    os.write(buffer, 0, count);
9712482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                }
9722482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            } catch (IOException e) {
9732482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov                throw new RuntimeException(e);
9742482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            }
9752482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov            mTestPhoto = os.toByteArray();
976a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov        }
9772482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        return mTestPhoto;
978a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov    }
979b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
980b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    public static void dump(ContentResolver resolver, boolean aggregatedOnly) {
981b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
982b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts._ID,
983b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts.DISPLAY_NAME
984b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
985b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String selection = null;
986b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        if (aggregatedOnly) {
987b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            selection = Contacts._ID
988b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                    + " IN (SELECT contact_id" +
989b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                    		" FROM raw_contacts GROUP BY contact_id HAVING count(*) > 1)";
990b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
991b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
992b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(Contacts.CONTENT_URI, projection, selection, null,
993b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts.DISPLAY_NAME);
994b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
995b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            long contactId = c.getLong(0);
996b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("Contact   ", String.format("%5d %s", contactId, c.getString(1)));
997b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            dumpRawContacts(resolver, contactId);
998b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("          ", ".");
999b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
1000b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
1001b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
1002b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
1003b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    private static void dumpRawContacts(ContentResolver resolver, long contactId) {
1004b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
1005b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                RawContacts._ID,
1006b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
1007b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(RawContacts.CONTENT_URI, projection, RawContacts.CONTACT_ID + "="
1008b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                + contactId, null, null);
1009b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
1010b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            long rawContactId = c.getLong(0);
1011b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("RawContact", String.format("      %-5d", rawContactId));
1012b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            dumpData(resolver, rawContactId);
1013b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
1014b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
1015b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
1016b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
1017b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    private static void dumpData(ContentResolver resolver, long rawContactId) {
1018b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
1019b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.MIMETYPE,
1020b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA1,
1021b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA2,
1022b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA3,
1023b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
1024b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(Data.CONTENT_URI, projection, Data.RAW_CONTACT_ID + "="
1025b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                + rawContactId, null, Data.MIMETYPE);
1026b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
1027b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            String mimetype = c.getString(0);
1028b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            if (Photo.CONTENT_ITEM_TYPE.equals(mimetype)) {
1029b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Log.i("Photo     ", "");
1030b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            } else {
1031b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                mimetype = mimetype.substring(mimetype.indexOf('/') + 1);
1032b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Log.i("Data      ", String.format("            %-10s %s,%s,%s", mimetype,
1033b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                        c.getString(1), c.getString(2), c.getString(3)));
1034b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            }
1035b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
1036b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
1037b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
1038285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov
103981d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov    protected void assertNetworkNotified(boolean expected) {
10409a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov        assertEquals(expected, (getContactsProvider()).isNetworkNotified());
104181d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov    }
10425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
104309ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov    protected void assertProjection(Uri uri, String[] expectedProjection) {
104409ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov        Cursor cursor = mResolver.query(uri, null, "0", null, null);
104509ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov        String[] actualProjection = cursor.getColumnNames();
104609ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov        MoreAsserts.assertEquals("Incorrect projection for URI: " + uri,
104709ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov                Sets.newHashSet(expectedProjection), Sets.newHashSet(actualProjection));
104809ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov        cursor.close();
104909ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov    }
105009ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov
10515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    /**
10525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * A contact in the database, and the attributes used to create it.  Construct using
10535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * {@link GoldenContactBuilder#build()}.
10545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     */
10555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    public final class GoldenContact {
10565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final long rawContactId;
10585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final long contactId;
10605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String givenName;
10625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10635b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String familyName;
10645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String nickname;
10665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final byte[] photo;
10685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String company;
10705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String title;
10725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String phone;
10745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String email;
10765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private GoldenContact(GoldenContactBuilder builder, long rawContactId, long contactId) {
10785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            this.rawContactId = rawContactId;
10805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            this.contactId = contactId;
10815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            givenName = builder.givenName;
10825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            familyName = builder.familyName;
10835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            nickname = builder.nickname;
10845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            photo = builder.photo;
10855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            company = builder.company;
10865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            title = builder.title;
10875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            phone = builder.phone;
10885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            email = builder.email;
10895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public void delete() {
10925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            Uri rawContactUri = ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId);
10935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.delete(rawContactUri, null, null);
10945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
10955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
10965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
10975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the index of the contact in table "raw_contacts"
10985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
10995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public long getRawContactId() {
11005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return rawContactId;
11015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the index of the contact in table "contacts"
11055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public long getContactId() {
11075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return contactId;
11085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11112d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill         * Returns the lookup key for the contact.
11122d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill         */
11132d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        public String getLookupKey() {
11142d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill            return queryLookupKey(contactId);
11152d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        }
11162d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill
11172d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        /**
11185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's given name.
11195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getGivenName() {
11215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return givenName;
11225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's family name.
11265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getFamilyName() {
11285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return familyName;
11295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11315b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11325b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's nickname.
11335b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11345b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getNickname() {
11355b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return nickname;
11365b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11375b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Return's the contact's photo
11405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11415b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public byte[] getPhoto() {
11425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return photo;
11435b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11445b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11455b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11465b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Return's the company at which the contact works.
11475b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getCompany() {
11495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return company;
11505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's job title.
11545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getTitle() {
11565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return title;
11575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's phone number
11615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getPhone() {
11635b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return phone;
11645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's email address
11685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
11695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getEmail() {
11705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return email;
11715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
11725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     }
11735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    /**
11755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * Builds {@link GoldenContact} objects.  Unspecified boolean objects default to false.
11765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * Unspecified String objects default to null.
11775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     */
11785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    public final class GoldenContactBuilder {
11795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String givenName;
11815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String familyName;
11835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String nickname;
11855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private byte[] photo;
11875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String company;
11895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String title;
11915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String phone;
11935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String email;
11955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
11975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's given and family names.
11985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         *
11995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * TODO(dplotnikov): inline, or should we require them to set both names if they set either?
12005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder name(String givenName, String familyName) {
12025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return givenName(givenName).familyName(familyName);
12035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's given name.
12075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder givenName(String value) {
12095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            givenName = value;
12105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
12115b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12125b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12135b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12145b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's family name.
12155b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12165b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder familyName(String value) {
12175b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            familyName = value;
12185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
12195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's nickname.
12235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder nickname(String value) {
12255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            nickname = value;
12265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
12275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's photo.
12315b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12325b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder photo(byte[] value) {
12335b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            photo = value;
12345b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
12355b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12365b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12375b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The company at which the contact works.
12395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder company(String value) {
12415b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            company = value;
12425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
12435b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12445b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12455b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12465b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's job title.
12475b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder title(String value) {
12495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            title = value;
12505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
12515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's phone number.
12555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder phone(String value) {
12575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            phone = value;
12585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
12595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's email address; also sets their IM status to {@link StatusUpdates#OFFLINE}
12635b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * with a presence of "Coding for Android".
12645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder email(String value) {
12665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            email = value;
12675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
12685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Builds the {@link GoldenContact} specified by this builder.
12725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContact build() {
12745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            final long groupId = createGroup(mAccount, "gsid1", "title1");
12765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            long rawContactId = createRawContact();
12785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            insertGroupMembership(rawContactId, groupId);
12795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (givenName != null || familyName != null) {
12815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertStructuredName(rawContactId, givenName, familyName);
12825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
12835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (nickname != null) {
12845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertNickname(rawContactId, nickname);
12855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
12865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (photo != null) {
12875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertPhoto(rawContactId);
12885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
12895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (company != null || title != null) {
12905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertOrganization(rawContactId);
12915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
12925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (email != null) {
12935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertEmail(rawContactId);
12945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
12955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (phone != null) {
12965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertPhone(rawContactId);
12975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
12985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            long contactId = queryContactId(rawContactId);
13005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return new GoldenContact(this, rawContactId, contactId);
13025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertPhoto(long rawContactId) {
13055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
13065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
13075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Photo.CONTENT_ITEM_TYPE);
13085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Photo.PHOTO, photo);
13095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
13105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13115b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13125b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertOrganization(long rawContactId) {
13135b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13145b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
13155b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
13165b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Organization.CONTENT_ITEM_TYPE);
13175b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Organization.TYPE, Organization.TYPE_WORK);
13185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (company != null) {
13195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                values.put(Organization.COMPANY, company);
13205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
13215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (title != null) {
13225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                values.put(Organization.TITLE, title);
13235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
13245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
13255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertEmail(long rawContactId) {
13285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
13305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
13315b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Email.CONTENT_ITEM_TYPE);
13325b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Email.TYPE, Email.TYPE_WORK);
13335b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Email.DATA, "foo@acme.com");
13345b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
13355b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13365b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            int protocol = Im.PROTOCOL_GOOGLE_TALK;
13375b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.clear();
13395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PROTOCOL, protocol);
13405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.IM_HANDLE, email);
13415b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.IM_ACCOUNT, "foo");
13425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PRESENCE_STATUS, StatusUpdates.OFFLINE);
1343aabcd1d34a71ad06ee0a9395331540484f1ceb17Vasu Nori            values.put(StatusUpdates.CHAT_CAPABILITY, StatusUpdates.CAPABILITY_HAS_CAMERA);
13445b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PRESENCE_CUSTOM_STATUS, "Coding for Android");
13455b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(StatusUpdates.CONTENT_URI, values);
13465b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13475b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertPhone(long rawContactId) {
13495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
13505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
13515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE);
13525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.IS_PRIMARY, 1);
13535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Phone.TYPE, Phone.TYPE_HOME);
13545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Phone.NUMBER, phone);
13555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
13565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    }
1358d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov}
1359