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
190bf6b318e3c994294d4a885f57906debd4a0e64eDaniel Lehmannimport static com.android.providers.contacts.ContactsActor.PACKAGE_GREY;
200bf6b318e3c994294d4a885f57906debd4a0e64eDaniel Lehmann
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;
30d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.database.Cursor;
31d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.net.Uri;
326f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoroimport android.os.Bundle;
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;
506f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoroimport android.provider.ContactsContract.SearchSnippetColumns;
5109ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.provider.ContactsContract.Settings;
5209ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.provider.ContactsContract.StatusUpdates;
533b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmannimport android.provider.ContactsContract.StreamItems;
5409ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport android.test.MoreAsserts;
55d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikovimport android.test.mock.MockContentResolver;
56b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikovimport android.util.Log;
57bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
589261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport java.util.ArrayList;
599261b2141aa90a4fed632fd6da03026d4c216280Fred Quintanaimport java.util.Arrays;
60e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikovimport java.util.Comparator;
61ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikovimport java.util.Iterator;
6220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikovimport java.util.Map;
63ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikovimport java.util.Map.Entry;
6409ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikovimport java.util.Set;
65d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
66d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov/**
67d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov * A common superclass for {@link ContactsProvider2}-related tests.
68d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov */
69f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoropublic abstract class BaseContactsProvider2Test extends PhotoLoadingTestCase {
70d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
71d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected static final String PACKAGE = "ContactsProvider2Test";
7273f1f396c155b247b903d8f4111db17d3e13dc4dDmitri Plotnikov    public static final String READ_ONLY_ACCOUNT_TYPE =
7373f1f396c155b247b903d8f4111db17d3e13dc4dDmitri Plotnikov            SynchronousContactsProvider2.READ_ONLY_ACCOUNT_TYPE;
74d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
75bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected ContactsActor mActor;
76d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected MockContentResolver mResolver;
779261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected Account mAccount = new Account("account1", "account type1");
78e2579e029472f76b2dfda141444d775c67da0ec8Cynthia Wong    protected Account mAccountTwo = new Account("account2", "account type2");
799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static Long NO_LONG = new Long(0);
819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static String NO_STRING = new String("");
829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected final static Account NO_ACCOUNT = new Account("a", "b");
83d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
84bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected Class<? extends ContentProvider> getProviderClass() {
85bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        return SynchronousContactsProvider2.class;
86bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    }
87bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
88bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    protected String getAuthority() {
89bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        return ContactsContract.AUTHORITY;
90bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov    }
91bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov
92d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    @Override
93d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    protected void setUp() throws Exception {
94d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        super.setUp();
95d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
96bee1a6b2e7cbea778195890e442c9e50f2a4e6d9Dmitri Plotnikov        mActor = new ContactsActor(getContext(), PACKAGE_GREY, getProviderClass(), getAuthority());
97d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        mResolver = mActor.resolver;
98445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov        if (mActor.provider instanceof SynchronousContactsProvider2) {
999a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov            getContactsProvider().wipeData();
100445ca81effbb0d61c7bc0033acf2d3dfd228fd4eDmitri Plotnikov        }
10124c1d384b45a6d3c1cc959062a9d4308335fabbfDave Santoro
10224c1d384b45a6d3c1cc959062a9d4308335fabbfDave Santoro        // Give the actor access to read/write contacts and profile data by default.
10324c1d384b45a6d3c1cc959062a9d4308335fabbfDave Santoro        mActor.addPermissions(
10424c1d384b45a6d3c1cc959062a9d4308335fabbfDave Santoro                "android.permission.READ_CONTACTS",
10524c1d384b45a6d3c1cc959062a9d4308335fabbfDave Santoro                "android.permission.WRITE_CONTACTS",
10636612112760df799ef89f7e324e5dfabd5ca0d2bDave Santoro                "android.permission.READ_SOCIAL_STREAM",
10736612112760df799ef89f7e324e5dfabd5ca0d2bDave Santoro                "android.permission.WRITE_SOCIAL_STREAM",
10824c1d384b45a6d3c1cc959062a9d4308335fabbfDave Santoro                "android.permission.READ_PROFILE",
10924c1d384b45a6d3c1cc959062a9d4308335fabbfDave Santoro                "android.permission.WRITE_PROFILE");
1109261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
1119261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
1120265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    @Override
1130265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    protected void tearDown() throws Exception {
1140265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov        super.tearDown();
1150265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov    }
1160265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov
1179a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov    public SynchronousContactsProvider2 getContactsProvider() {
1189a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov        return (SynchronousContactsProvider2) mActor.provider;
1199a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov    }
1209a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov
121e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public Context getMockContext() {
122e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        return mActor.context;
123e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
124e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
125e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public void addAuthority(String authority) {
126e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        mActor.addAuthority(authority);
127e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
128e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
129e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    public ContentProvider addProvider(Class<? extends ContentProvider> providerClass,
130e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov            String authority) throws Exception {
131e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov        return mActor.addProvider(providerClass, authority);
132e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov    }
133e99988b266dd1263162583e81e2b408e7329b1c8Dmitri Plotnikov
1343d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov    public ContentProvider getProvider() {
1353d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov        return mActor.provider;
1363d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov    }
1373d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov
1389261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected Uri maybeAddAccountQueryParameters(Uri uri, Account account) {
1399261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == null) {
1409261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return uri;
1419261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
1429261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return uri.buildUpon()
143df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana                .appendQueryParameter(RawContacts.ACCOUNT_NAME, account.name)
144df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana                .appendQueryParameter(RawContacts.ACCOUNT_TYPE, account.type)
1459261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                .build();
146d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
147d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
148d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long createRawContact() {
149226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        return createRawContact(null);
150d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
151d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
152c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov    protected long createRawContactWithName() {
153e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName(null);
154e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
155e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
156e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected long createRawContactWithName(Account account) {
157e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName("John", "Doe", account);
1583cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov    }
1593cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov
1603cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov    protected long createRawContactWithName(String firstName, String lastName) {
161e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return createRawContactWithName(firstName, lastName, null);
162e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
163e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
164e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected long createRawContactWithName(String firstName, String lastName, Account account) {
165e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        long rawContactId = createRawContact(account);
1663cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        insertStructuredName(rawContactId, firstName, lastName);
167c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        return rawContactId;
168c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov    }
169c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov
170e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    protected Uri setCallerIsSyncAdapter(Uri uri, Account account) {
171e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        if (account == null) {
172e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey            return uri;
173e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        }
174e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        final Uri.Builder builder = uri.buildUpon();
175e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.RawContacts.ACCOUNT_NAME, account.name);
176e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.RawContacts.ACCOUNT_TYPE, account.type);
177e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        builder.appendQueryParameter(ContactsContract.CALLER_IS_SYNCADAPTER, "true");
178e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey        return builder.build();
179e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey    }
180e246689441b2ff39cb97de277d6caeec95358863Jeff Sharkey
181226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana    protected long createRawContact(Account account, String... extras) {
1829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
183dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        extrasVarArgsToValues(values, extras);
184dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        final Uri uri = maybeAddAccountQueryParameters(RawContacts.CONTENT_URI, account);
185dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        Uri contactUri = mResolver.insert(uri, values);
186dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return ContentUris.parseId(contactUri);
187dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
188dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
189dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected int updateItem(Uri uri, long id, String... extras) {
190dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        Uri itemUri = ContentUris.withAppendedId(uri, id);
191dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return updateItem(itemUri, extras);
192dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
193dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
194dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected int updateItem(Uri uri, String... extras) {
195dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        ContentValues values = new ContentValues();
196dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        extrasVarArgsToValues(values, extras);
197dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return mResolver.update(uri, values, null, null);
198dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
199dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
200dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    private static void extrasVarArgsToValues(ContentValues values, String... extras) {
201226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        for (int i = 0; i < extras.length; ) {
202226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana            values.put(extras[i], extras[i + 1]);
203226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana            i += 2;
204226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana        }
2059261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
2069261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
2079261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long createGroup(Account account, String sourceId, String title) {
208dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return createGroup(account, sourceId, title, 1, false, false);
2090be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov    }
2100be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov
2110be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov    protected long createGroup(Account account, String sourceId, String title, int visible) {
212dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        return createGroup(account, sourceId, title, visible, false, false);
213dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
214dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
215f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov    protected long createAutoAddGroup(Account account) {
216f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov        return createGroup(account, "auto", "auto",
217f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov                0 /* visible */,  true /* auto-add */, false /* fav */);
218f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov    }
219f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov
220dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected long createGroup(Account account, String sourceId, String title,
221dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            int visible, boolean autoAdd, boolean favorite) {
2229261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
2239261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Groups.SOURCE_ID, sourceId);
2249261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Groups.TITLE, title);
2250be993f8ef0078b9825a5ffe6add08a6786d8dacDmitri Plotnikov        values.put(Groups.GROUP_VISIBLE, visible);
226dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        values.put(Groups.AUTO_ADD, autoAdd ? 1 : 0);
227dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        values.put(Groups.FAVORITES, favorite ? 1 : 0);
2289261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        final Uri uri = maybeAddAccountQueryParameters(Groups.CONTENT_URI, account);
2299261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return ContentUris.parseId(mResolver.insert(uri, values));
2309261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
2319261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
23289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected void createSettings(Account account, String shouldSync, String ungroupedVisible) {
233f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro        createSettings(new AccountWithDataSet(account.name, account.type, null),
234f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro                shouldSync, ungroupedVisible);
235f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro    }
236f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro
237f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro    protected void createSettings(AccountWithDataSet account, String shouldSync,
238f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro            String ungroupedVisible) {
23989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        ContentValues values = new ContentValues();
240f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro        values.put(Settings.ACCOUNT_NAME, account.getAccountName());
241f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro        values.put(Settings.ACCOUNT_TYPE, account.getAccountType());
242f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro        if (account.getDataSet() != null) {
243f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro            values.put(Settings.DATA_SET, account.getDataSet());
244f9b77edaf5855bf6932fbc4b4b4342273669efefDave Santoro        }
24589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.SHOULD_SYNC, shouldSync);
24689c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        values.put(Settings.UNGROUPED_VISIBLE, ungroupedVisible);
24789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        mResolver.insert(Settings.CONTENT_URI, values);
24889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
24989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
2505ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertStructuredName(long rawContactId, String givenName, String familyName) {
251d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        ContentValues values = new ContentValues();
252d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        StringBuilder sb = new StringBuilder();
253d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (givenName != null) {
254d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(givenName);
255d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
256d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (givenName != null && familyName != null) {
257d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(" ");
258d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
259d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        if (familyName != null) {
260d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov            sb.append(familyName);
261d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        }
262d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.DISPLAY_NAME, sb.toString());
263d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.GIVEN_NAME, givenName);
264d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(StructuredName.FAMILY_NAME, familyName);
265d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
2665ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        return insertStructuredName(rawContactId, values);
2674097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov    }
2684097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
2695ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertStructuredName(long rawContactId, ContentValues values) {
2705ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
2714097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        values.put(Data.MIMETYPE, StructuredName.CONTENT_ITEM_TYPE);
272d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
273d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        return resultUri;
274d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
275d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
276a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka    protected Uri insertOrganization(long rawContactId, ContentValues values) {
277a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka        return insertOrganization(rawContactId, values, false);
27801911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    }
27901911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
280a5d05d90333a70d471d78e82caeb5cfa2e4d4c59Tadashi G. Takaoka    protected Uri insertOrganization(long rawContactId, ContentValues values, boolean primary) {
28101911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
28201911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Data.MIMETYPE, Organization.CONTENT_ITEM_TYPE);
28301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        values.put(Organization.TYPE, Organization.TYPE_WORK);
28401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        if (primary) {
2850265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
28601911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        }
28701911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
28801911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
28901911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        return resultUri;
29001911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov    }
29101911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov
2925ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber) {
29325abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        return insertPhoneNumber(rawContactId, phoneNumber, false);
29425abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    }
29525abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov
29625abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber, boolean primary) {
29758567abca253f1efa2db5c39e17e42dca589e916Daisuke Miyakawa        return insertPhoneNumber(rawContactId, phoneNumber, primary, Phone.TYPE_HOME);
29858567abca253f1efa2db5c39e17e42dca589e916Daisuke Miyakawa    }
29958567abca253f1efa2db5c39e17e42dca589e916Daisuke Miyakawa
30058567abca253f1efa2db5c39e17e42dca589e916Daisuke Miyakawa    protected Uri insertPhoneNumber(long rawContactId, String phoneNumber, boolean primary,
30158567abca253f1efa2db5c39e17e42dca589e916Daisuke Miyakawa            int type) {
302bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        ContentValues values = new ContentValues();
3035ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
304bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        values.put(Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE);
305bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        values.put(Phone.NUMBER, phoneNumber);
30658567abca253f1efa2db5c39e17e42dca589e916Daisuke Miyakawa        values.put(Phone.TYPE, type);
30725abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        if (primary) {
3080265a180cf027d149f11f8750652ac67ea08ca24Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
30925abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        }
310bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
311bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
312bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov        return resultUri;
313bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov    }
314bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
3155ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email) {
31625abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        return insertEmail(rawContactId, email, false);
31725abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    }
31825abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov
31925abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email, boolean primary) {
320916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        return insertEmail(rawContactId, email, primary, Email.TYPE_HOME, null);
321916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov    }
322916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov
323916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov    protected Uri insertEmail(long rawContactId, String email, boolean primary, int type,
324916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov            String label) {
325a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        ContentValues values = new ContentValues();
3265ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
327a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Data.MIMETYPE, Email.CONTENT_ITEM_TYPE);
328a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Email.DATA, email);
329916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        values.put(Email.TYPE, type);
330916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov        values.put(Email.LABEL, label);
33125abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        if (primary) {
332916f2d7104bfba857412a66b40ed60fea6546222Dmitri Plotnikov            values.put(Data.IS_PRIMARY, 1);
33325abcf949c0dd826a770b437489b83de48975ceaDmitri Plotnikov        }
334a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
335a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
336a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        return resultUri;
337a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov    }
338a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
3395ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertNickname(long rawContactId, String nickname) {
340a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        ContentValues values = new ContentValues();
3415ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
342a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Data.MIMETYPE, Nickname.CONTENT_ITEM_TYPE);
343a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        values.put(Nickname.NAME, nickname);
3443cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Nickname.TYPE, Nickname.TYPE_OTHER_NAME);
345a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
346a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
347a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov        return resultUri;
348a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov    }
349a5ad551e1753086825499f1aeb6415bb986f3588Dmitri Plotnikov
3504a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov    protected Uri insertPostalAddress(long rawContactId, String formattedAddress) {
3514a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        ContentValues values = new ContentValues();
3524a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3534a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(Data.MIMETYPE, StructuredPostal.CONTENT_ITEM_TYPE);
3544a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        values.put(StructuredPostal.FORMATTED_ADDRESS, formattedAddress);
3554a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov
3564a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
3574a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov        return resultUri;
3584a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov    }
3594a023070dab9a069be4cac5f5ba5554b66238484Dmitri Plotnikov
360eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov    protected Uri insertPostalAddress(long rawContactId, ContentValues values) {
361eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
362eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov        values.put(Data.MIMETYPE, StructuredPostal.CONTENT_ITEM_TYPE);
363eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
364eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov        return resultUri;
365eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov    }
366eeeed5669d98897501bf2b18c88579c7effd0955Dmitri Plotnikov
3675ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertPhoto(long rawContactId) {
368ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        ContentValues values = new ContentValues();
3695ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
370ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        values.put(Data.MIMETYPE, Photo.CONTENT_ITEM_TYPE);
3712482dd0309794d57233d0af37a7c12fc647e7bf9Dmitri Plotnikov        values.put(Photo.PHOTO, loadTestPhoto());
372ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
373ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        return resultUri;
374ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
375ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
376f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro    protected Uri insertPhoto(long rawContactId, int resourceId) {
377f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro        ContentValues values = new ContentValues();
378f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro        values.put(Data.RAW_CONTACT_ID, rawContactId);
379f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro        values.put(Data.MIMETYPE, Photo.CONTENT_ITEM_TYPE);
380f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro        values.put(Photo.PHOTO, loadPhotoFromResource(resourceId, PhotoSize.ORIGINAL));
381f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
382f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro        return resultUri;
383f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro    }
384f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro
3855ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertGroupMembership(long rawContactId, String sourceId) {
3869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
3875ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3889261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Data.MIMETYPE, GroupMembership.CONTENT_ITEM_TYPE);
3899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(GroupMembership.GROUP_SOURCE_ID, sourceId);
3909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return mResolver.insert(Data.CONTENT_URI, values);
3919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
3929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
3935ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected Uri insertGroupMembership(long rawContactId, Long groupId) {
3949261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
3955ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
3969261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(Data.MIMETYPE, GroupMembership.CONTENT_ITEM_TYPE);
3979261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put(GroupMembership.GROUP_ROW_ID, groupId);
3989261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return mResolver.insert(Data.CONTENT_URI, values);
3999261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
4009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
401f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov    public void removeGroupMemberships(long rawContactId) {
402f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov        mResolver.delete(Data.CONTENT_URI,
403f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov                Data.MIMETYPE + "=? AND " + GroupMembership.RAW_CONTACT_ID + "=?",
404f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov                new String[] { GroupMembership.CONTENT_ITEM_TYPE, String.valueOf(rawContactId) });
405f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov    }
406f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov
40782bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov    protected Uri insertStatusUpdate(int protocol, String customProtocol, String handle,
408aabcd1d34a71ad06ee0a9395331540484f1ceb17Vasu Nori            int presence, String status, int chatMode) {
4095d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro        return insertStatusUpdate(protocol, customProtocol, handle, presence, status, chatMode,
4105d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro                false);
411a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov    }
412a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov
41382bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov    protected Uri insertStatusUpdate(int protocol, String customProtocol, String handle,
4145d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro            int presence, String status, int chatMode, boolean isUserProfile) {
4155d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro        return insertStatusUpdate(protocol, customProtocol, handle, presence, status, 0, chatMode,
4165d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro                isUserProfile);
4175d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro    }
4185d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro
4195d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro    protected Uri insertStatusUpdate(int protocol, String customProtocol, String handle,
4205d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro            int presence, String status, long timestamp, int chatMode, boolean isUserProfile) {
421bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        ContentValues values = new ContentValues();
42282bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.PROTOCOL, protocol);
42382bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.CUSTOM_PROTOCOL, customProtocol);
42482bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov        values.put(StatusUpdates.IM_HANDLE, handle);
4255d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro        return insertStatusUpdate(values, presence, status, timestamp, chatMode, isUserProfile);
426f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    }
427f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov
428f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    protected Uri insertStatusUpdate(
429f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov            long dataId, int presence, String status, long timestamp, int chatMode) {
4305d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro        return insertStatusUpdate(dataId, presence, status, timestamp, chatMode, false);
4315d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro    }
4325d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro
4335d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro    protected Uri insertStatusUpdate(
4345d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro            long dataId, int presence, String status, long timestamp, int chatMode,
4355d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro            boolean isUserProfile) {
436f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov        ContentValues values = new ContentValues();
437f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov        values.put(StatusUpdates.DATA_ID, dataId);
4385d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro        return insertStatusUpdate(values, presence, status, timestamp, chatMode, isUserProfile);
439f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    }
440f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov
441f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    private Uri insertStatusUpdate(
4425d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro            ContentValues values, int presence, String status, long timestamp, int chatMode,
4435d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro            boolean isUserProfile) {
444a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (presence != 0) {
44582bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.PRESENCE, presence);
446aabcd1d34a71ad06ee0a9395331540484f1ceb17Vasu Nori            values.put(StatusUpdates.CHAT_CAPABILITY, chatMode);
447a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
448a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (status != null) {
44982bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.STATUS, status);
450a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
451a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        if (timestamp != 0) {
45282bd858c9911dfbd8dca52dc276333768b0a429eDmitri Plotnikov            values.put(StatusUpdates.STATUS_TIMESTAMP, timestamp);
453a23cd5b6f478f6c9dda54173e84bd0098b9f3364Dmitri Plotnikov        }
454bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
4555d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro        Uri insertUri = isUserProfile
4565d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro                ? StatusUpdates.PROFILE_CONTENT_URI
4575d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro                : StatusUpdates.CONTENT_URI;
4585d0a768b56ed4bd0dfef81b8389247ba74766659Dave Santoro        Uri resultUri = mResolver.insert(insertUri, values);
459bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        return resultUri;
460bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
461bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
4623b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann    protected Uri insertStreamItem(long rawContactId, ContentValues values, Account account) {
4633b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann        return mResolver.insert(
4643b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                maybeAddAccountQueryParameters(
4653b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                        Uri.withAppendedPath(
4663b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                                ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
4673b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                                RawContacts.StreamItems.CONTENT_DIRECTORY),
4683b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                        account),
4693b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                values);
4703b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann    }
4713b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann
4723b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann    protected Uri insertStreamItemPhoto(long streamItemId, ContentValues values, Account account) {
4733b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann        return mResolver.insert(
4743b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                maybeAddAccountQueryParameters(
4753b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                        Uri.withAppendedPath(
4763b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                                ContentUris.withAppendedId(StreamItems.CONTENT_URI, streamItemId),
4773b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                                StreamItems.StreamItemPhotos.CONTENT_DIRECTORY),
4783b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                        account),
4793b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann                values);
4803b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann    }
4813b505de6c622d20d40b85b361c1437a89aef82deDaniel Lehmann
4824dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov    protected Uri insertImHandle(long rawContactId, int protocol, String customProtocol,
4834dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov            String handle) {
484bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        ContentValues values = new ContentValues();
4855ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
486bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Data.MIMETYPE, Im.CONTENT_ITEM_TYPE);
487bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Im.PROTOCOL, protocol);
4884dcd106ccc27dbbfaae86baf0cd57beb42c27cccDmitri Plotnikov        values.put(Im.CUSTOM_PROTOCOL, customProtocol);
489bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        values.put(Im.DATA, handle);
4903cebbf7141252768d3e272e049e9c5b0cb9d710eDmitri Plotnikov        values.put(Im.TYPE, Im.TYPE_HOME);
491bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
492bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
493bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar        return resultUri;
494bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
495bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
4966d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov    protected Uri insertEvent(long rawContactId, int type, String date) {
4976d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        ContentValues values = new ContentValues();
4986d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
4996d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Data.MIMETYPE, Event.CONTENT_ITEM_TYPE);
5006d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Event.TYPE, type);
5016d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Event.START_DATE, date);
5026d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
5036d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        return resultUri;
5046d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov    }
5056d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov
5066d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov    protected Uri insertNote(long rawContactId, String note) {
5076d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        ContentValues values = new ContentValues();
5086d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Data.RAW_CONTACT_ID, rawContactId);
5096d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Data.MIMETYPE, Note.CONTENT_ITEM_TYPE);
5106d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        values.put(Note.NOTE, note);
5116d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        Uri resultUri = mResolver.insert(Data.CONTENT_URI, values);
5126d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov        return resultUri;
5136d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov    }
5146d9702cec82fd27a1c3093c64df9dcc22744899aDmitri Plotnikov
515767e109d6f08749b9ed202c0b71f3459eaae2115Dmitri Plotnikov    protected void setContactAccount(long rawContactId, String accountType, String accountName) {
516ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        ContentValues values = new ContentValues();
517767e109d6f08749b9ed202c0b71f3459eaae2115Dmitri Plotnikov        values.put(RawContacts.ACCOUNT_TYPE, accountType);
5186cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov        values.put(RawContacts.ACCOUNT_NAME, accountName);
519ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
520ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        mResolver.update(ContentUris.withAppendedId(
5215ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov                RawContacts.CONTENT_URI, rawContactId), values, null, null);
522ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
523ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
5240c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov    protected void setAggregationException(int type, long rawContactId1, long rawContactId2) {
525d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        ContentValues values = new ContentValues();
5260c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        values.put(AggregationExceptions.RAW_CONTACT_ID1, rawContactId1);
5270c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        values.put(AggregationExceptions.RAW_CONTACT_ID2, rawContactId2);
528d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        values.put(AggregationExceptions.TYPE, type);
5293d8b043c3341a5b6c2e781b7eba9767d5cd13267Dmitri Plotnikov        assertEquals(1, mResolver.update(AggregationExceptions.CONTENT_URI, values, null, null));
530d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
531d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
532d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected Cursor queryRawContact(long rawContactId) {
5330c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
5340c0adda32be5de3acf392ab715cff468b6b340f8Dmitri Plotnikov                null, null, null, null);
535d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
536d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
537d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected Cursor queryContact(long contactId) {
538d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
539d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov                null, null, null, null);
540d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
541d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
542af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov    protected Cursor queryContact(long contactId, String[] projection) {
543af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov        return mResolver.query(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
544af088aeb51685eed17580edc04b495d12232ecf9Dmitri Plotnikov                projection, null, null, null);
545bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar    }
546bc5c799a52b5bde2f273efd118ebe2228c3d8f15Evan Millar
547f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    protected Uri getContactUriForRawContact(long rawContactId) {
548f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov        return ContentUris.withAppendedId(Contacts.CONTENT_URI, queryContactId(rawContactId));
549f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov    }
550f4015ab9ab7c26b766b5331fbf6655b8c54877eaDmitri Plotnikov
551d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long queryContactId(long rawContactId) {
552d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryRawContact(rawContactId);
553d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertTrue(c.moveToFirst());
554d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId = c.getLong(c.getColumnIndex(RawContacts.CONTACT_ID));
555d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        c.close();
556d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        return contactId;
557d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
558d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
559d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected long queryPhotoId(long contactId) {
560d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryContact(contactId);
561ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        assertTrue(c.moveToFirst());
562d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long photoId = c.getInt(c.getColumnIndex(Contacts.PHOTO_ID));
563ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        c.close();
564ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar        return photoId;
565ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar    }
566ae6ca1f34cf5458d79ec803411d4308879a91e92Evan Millar
56787614d7d293b1519dc1f0f403fd59c8bf4d8a347Dave Santoro    protected long queryPhotoFileId(long contactId) {
56887614d7d293b1519dc1f0f403fd59c8bf4d8a347Dave Santoro        return getStoredLongValue(ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId),
56987614d7d293b1519dc1f0f403fd59c8bf4d8a347Dave Santoro                Contacts.PHOTO_FILE_ID);
57087614d7d293b1519dc1f0f403fd59c8bf4d8a347Dave Santoro    }
57187614d7d293b1519dc1f0f403fd59c8bf4d8a347Dave Santoro
572dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected boolean queryRawContactIsStarred(long rawContactId) {
573dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        Cursor c = queryRawContact(rawContactId);
574dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        try {
575dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            assertTrue(c.moveToFirst());
576dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            return c.getLong(c.getColumnIndex(RawContacts.STARRED)) != 0;
577dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        } finally {
578dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            c.close();
579dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        }
580dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
581dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
582d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected String queryDisplayName(long contactId) {
583d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        Cursor c = queryContact(contactId);
584d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertTrue(c.moveToFirst());
585d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        String displayName = c.getString(c.getColumnIndex(Contacts.DISPLAY_NAME));
586d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        c.close();
587d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        return displayName;
588d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
589d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
5902149ab82f021c204618d0d3644e261fd7a8d8490Dmitri Plotnikov    protected String queryLookupKey(long contactId) {
5912d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        Cursor c = queryContact(contactId);
5922d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        assertTrue(c.moveToFirst());
5932d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        String lookupKey = c.getString(c.getColumnIndex(Contacts.LOOKUP_KEY));
5942d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        c.close();
5952d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        return lookupKey;
5962d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill    }
5972d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill
598d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertAggregated(long rawContactId1, long rawContactId2) {
599d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
600d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
601d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 == contactId2);
602bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov    }
603bf659107617a6291ba8bfeebc3f2e50138075ab5Dmitri Plotnikov
604d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertAggregated(long rawContactId1, long rawContactId2,
605d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov            String expectedDisplayName) {
606d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
607d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
608d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 == contactId2);
609d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
610d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        String displayName = queryDisplayName(contactId1);
611d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov        assertEquals(expectedDisplayName, displayName);
612d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
613d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov
614d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov    protected void assertNotAggregated(long rawContactId1, long rawContactId2) {
615d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId1 = queryContactId(rawContactId1);
616d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        long contactId2 = queryContactId(rawContactId2);
617d3fde755e73cd3912a488e7cb7d412d3c5f6ca94Dmitri Plotnikov        assertTrue(contactId1 != contactId2);
618d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov    }
6194097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
6205ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected void assertStructuredName(long rawContactId, String prefix, String givenName,
6214097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov            String middleName, String familyName, String suffix) {
622f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov        Uri uri = Uri.withAppendedPath(
623f266e8c568905337960b1fec5379841585af92a7Dmitri Plotnikov                ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId),
6246cffee46a1334d2b3ed19f436b27638451541044Dmitri Plotnikov                RawContacts.Data.CONTENT_DIRECTORY);
6254097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
6264097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        final String[] projection = new String[] {
6274097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                StructuredName.PREFIX, StructuredName.GIVEN_NAME, StructuredName.MIDDLE_NAME,
6284097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                StructuredName.FAMILY_NAME, StructuredName.SUFFIX
6294097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        };
6304097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
6314097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        Cursor c = mResolver.query(uri, projection, Data.MIMETYPE + "='"
6324097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov                + StructuredName.CONTENT_ITEM_TYPE + "'", null, null);
6334097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov
6344097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertTrue(c.moveToFirst());
6354097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(prefix, c.getString(0));
6364097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(givenName, c.getString(1));
6374097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(middleName, c.getString(2));
6384097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(familyName, c.getString(3));
6394097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        assertEquals(suffix, c.getString(4));
6404097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov        c.close();
6414097855e2d672b3f8e1c5c8a169efb80203bf53eDmitri Plotnikov    }
6429261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6439261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long assertSingleGroup(Long rowId, Account account, String sourceId, String title) {
6449261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(Groups.CONTENT_URI, null, null, null, null);
6459261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        try {
6469261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.moveToNext());
6479261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long actualRowId = assertGroup(c, rowId, account, sourceId, title);
6489261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertFalse(c.moveToNext());
6499261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return actualRowId;
6509261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } finally {
6519261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            c.close();
6529261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6539261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6549261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6555ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected long assertSingleGroupMembership(Long rowId, Long rawContactId, Long groupRowId,
6569261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String sourceId) {
6579261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(ContactsContract.Data.CONTENT_URI, null, null, null, null);
6589261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        try {
6599261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.moveToNext());
6605ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov            long actualRowId = assertGroupMembership(c, rowId, rawContactId, groupRowId, sourceId);
6619261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertFalse(c.moveToNext());
6629261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return actualRowId;
6639261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } finally {
6649261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            c.close();
6659261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6669261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6679261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6685ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov    protected long assertGroupMembership(Cursor c, Long rowId, Long rawContactId, Long groupRowId,
6699261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String sourceId) {
6709261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, rowId, Data._ID);
6715ef0401c311c62e53bde415b99cbb0ff83b0a9a2Dmitri Plotnikov        assertNullOrEquals(c, rawContactId, GroupMembership.RAW_CONTACT_ID);
6729261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, groupRowId, GroupMembership.GROUP_ROW_ID);
6739261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, sourceId, GroupMembership.GROUP_SOURCE_ID);
6749261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return c.getLong(c.getColumnIndexOrThrow("_id"));
6759261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6769261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6779261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long assertGroup(Cursor c, Long rowId, Account account, String sourceId, String title) {
6789261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, rowId, Groups._ID);
6799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, account);
6809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, sourceId, Groups.SOURCE_ID);
6819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertNullOrEquals(c, title, Groups.TITLE);
6829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return c.getLong(c.getColumnIndexOrThrow("_id"));
6839261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6849261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6859261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, Account account) {
6869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == NO_ACCOUNT) {
6879261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return;
6889261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (account == null) {
6909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.isNull(c.getColumnIndexOrThrow(Groups.ACCOUNT_NAME)));
6919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            assertTrue(c.isNull(c.getColumnIndexOrThrow(Groups.ACCOUNT_TYPE)));
6929261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        } else {
693df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana            assertEquals(account.name, c.getString(c.getColumnIndexOrThrow(Groups.ACCOUNT_NAME)));
694df9fd6b239de5829b04cb413e4dfa3e6da649c38Fred Quintana            assertEquals(account.type, c.getString(c.getColumnIndexOrThrow(Groups.ACCOUNT_TYPE)));
6959261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
6969261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
6979261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
6989261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, Long value, String columnName) {
6999261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (value != NO_LONG) {
7009261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (value == null) assertTrue(c.isNull(c.getColumnIndexOrThrow(columnName)));
7019261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            else assertEquals((long) value, c.getLong(c.getColumnIndexOrThrow(columnName)));
7029261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
7039261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7049261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7059261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    private void assertNullOrEquals(Cursor c, String value, String columnName) {
7069261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        if (value != NO_STRING) {
7079261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (value == null) assertTrue(c.isNull(c.getColumnIndexOrThrow(columnName)));
7089261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            else assertEquals(value, c.getString(c.getColumnIndexOrThrow(columnName)));
7099261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
7109261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7119261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7129261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void assertDataRow(ContentValues actual, String expectedMimetype,
7139261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            Object... expectedArguments) {
7149261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertEquals(actual.toString(), expectedMimetype, actual.getAsString(Data.MIMETYPE));
7159261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        for (int i = 0; i < expectedArguments.length; i += 2) {
7169261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            String columnName = (String) expectedArguments[i];
7179261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            Object expectedValue = expectedArguments[i + 1];
7189261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue instanceof Uri) {
7199261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                expectedValue = ContentUris.parseId((Uri) expectedValue);
7209261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
7219261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue == null) {
7229261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana                assertNull(actual.toString(), actual.get(columnName));
7239261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
7249261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (expectedValue instanceof Long) {
725226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
726226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsLong(columnName));
7279261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else if (expectedValue instanceof Integer) {
728226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
729226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsInteger(columnName));
7309261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else if (expectedValue instanceof String) {
731226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
732226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.getAsString(columnName));
7339261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            } else {
734226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                assertEquals("mismatch at " + columnName + " from " + actual.toString(),
735226c3dc6e93ca76a84c99100caa31045cba06cf6Fred Quintana                        expectedValue, actual.get(columnName));
7369261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            }
7379261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
7389261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7399261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
740dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    protected void assertNoRowsAndClose(Cursor c) {
741dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        try {
742dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            assertFalse(c.moveToNext());
743dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        } finally {
744dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana            c.close();
745dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana        }
746dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana    }
747dd5c25c65f09ada246c826fb6d04f0b6d4cf4388Fred Quintana
7489261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected static class IdComparator implements Comparator<ContentValues> {
74959f6477e10203617f504857f7e9aee2fda393f4fDebashish Chatterjee        @Override
7509261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        public int compare(ContentValues o1, ContentValues o2) {
7519261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long id1 = o1.getAsLong(ContactsContract.Data._ID);
7529261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            long id2 = o2.getAsLong(ContactsContract.Data._ID);
7539261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            if (id1 == id2) return 0;
7549261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            return (id1 < id2) ? -1 : 1;
7559261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
7569261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7579261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7589261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected ContentValues[] asSortedContentValuesArray(
7599261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            ArrayList<Entity.NamedContentValues> subValues) {
7609261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues[] result = new ContentValues[subValues.size()];
7619261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        int i = 0;
7629261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        for (Entity.NamedContentValues subValue : subValues) {
7639261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            result[i] = subValue.values;
7649261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana            i++;
7659261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        }
7669261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Arrays.sort(result, new IdComparator());
7679261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return result;
7689261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7699261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7709261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void assertDirty(Uri uri, boolean state) {
7719261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(uri, new String[]{"dirty"}, null, null, null);
7729261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertTrue(c.moveToNext());
7739261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertEquals(state, c.getLong(0) != 0);
7749261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertFalse(c.moveToNext());
7753cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        c.close();
7769261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7779261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7789261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected long getVersion(Uri uri) {
7799261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        Cursor c = mResolver.query(uri, new String[]{"version"}, null, null, null);
7809261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertTrue(c.moveToNext());
7819261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        long version = c.getLong(0);
7829261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        assertFalse(c.moveToNext());
7833cfe8d532d509fbbe605454e3a32b2361b7e1501Dmitri Plotnikov        c.close();
7849261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        return version;
7859261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
7869261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana
7879261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    protected void clearDirty(Uri uri) {
7889261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        ContentValues values = new ContentValues();
7899261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        values.put("dirty", 0);
7909261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana        mResolver.update(uri, values, null, null);
7919261b2141aa90a4fed632fd6da03026d4c216280Fred Quintana    }
79220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
7935870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    protected void storeValue(Uri contentUri, long id, String column, String value) {
794f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        storeValue(ContentUris.withAppendedId(contentUri, id), column, value);
795f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
796f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
797f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, String column, String value) {
798f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        ContentValues values = new ContentValues();
799f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        values.put(column, value);
800f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
801f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        mResolver.update(contentUri, values, null, null);
802f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
803f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
804f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, long id, String column, long value) {
805f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        storeValue(ContentUris.withAppendedId(contentUri, id), column, value);
806f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    }
807f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov
808f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov    protected void storeValue(Uri contentUri, String column, long value) {
8095870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        ContentValues values = new ContentValues();
8105870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov        values.put(column, value);
8115870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov
812f5a847e8c93db02f3334dbc276debd90bdea5658Dmitri Plotnikov        mResolver.update(contentUri, values, null, null);
8135870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    }
8145870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov
815f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov    protected void assertStoredValue(Uri contentUri, long id, String column, Object expectedValue) {
816f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov        assertStoredValue(ContentUris.withAppendedId(contentUri, id), column, expectedValue);
817f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov    }
818f01c876a92b9c950a0450ed8b706ac5eb2c9b660Dmitri Plotnikov
8195870f2dcc2ac7715b2c078a886ee346622e7887eDmitri Plotnikov    protected void assertStoredValue(Uri rowUri, String column, Object expectedValue) {
82020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        String value = getStoredValue(rowUri, column);
82101911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        if (expectedValue == null) {
82201911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            assertNull("Column value " + column, value);
82301911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        } else {
82401911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov            assertEquals("Column value " + column, String.valueOf(expectedValue), value);
82501911fa9cfa21f198fd767eedde072acbb879f28Dmitri Plotnikov        }
82620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
82720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
82889c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected void assertStoredValue(Uri rowUri, String selection, String[] selectionArgs,
82989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            String column, Object expectedValue) {
83089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        String value = getStoredValue(rowUri, selection, selectionArgs, column);
83189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        if (expectedValue == null) {
83289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            assertNull("Column value " + column, value);
83389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        } else {
83489c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            assertEquals("Column value " + column, String.valueOf(expectedValue), value);
83589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        }
83689c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
83789c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
83820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected String getStoredValue(Uri rowUri, String column) {
83989c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        return getStoredValue(rowUri, null, null, column);
84089c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    }
84189c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov
84289c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov    protected String getStoredValue(Uri uri, String selection, String[] selectionArgs,
84389c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov            String column) {
8448e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov        String value = null;
84589c626eb655440c86a2e5df076e83708c1b32c17Dmitri Plotnikov        Cursor c = mResolver.query(uri, new String[] { column }, selection, selectionArgs, null);
84620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
847d1746e09bc7739f3d1449cececc66d5045ada498Daniel Lehmann            assertEquals("Record count for " + uri, 1, c.getCount());
84892ddc5cdc4d89ee2c6e861ae7b3a3a913ffa0100Dmitri Plotnikov
8498e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov            if (c.moveToFirst()) {
8506f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                value = getCursorStringValue(c, column);
8518e45e5f2142db78941b095f7418cc05b71668094Dmitri Plotnikov            }
85220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
85320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
85420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
85520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        return value;
85620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
85720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
8586f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro    /**
8596f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro     * Retrieves the string value in the given column, handling deferred snippeting if the requested
8606f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro     * column is the snippet and the cursor specifies it.
8616f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro     */
8626f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro    protected String getCursorStringValue(Cursor c, String column) {
8636f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro        String value = c.getString(c.getColumnIndex(column));
8646f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro        if (SearchSnippetColumns.SNIPPET.equals(column)) {
8656f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro            Bundle extras = c.getExtras();
8666f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro            if (extras.containsKey(ContactsContract.DEFERRED_SNIPPETING)) {
8676f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                String displayName = "No display name";
8686f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                int displayNameColumnIndex = c.getColumnIndex(Contacts.DISPLAY_NAME);
8696f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                if (displayNameColumnIndex != -1) {
8706f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                    displayName = c.getString(displayNameColumnIndex);
8716f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                }
8726f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                String query = extras.getString(ContactsContract.DEFERRED_SNIPPETING_QUERY);
8736f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                value = ContactsContract.snippetize(value, displayName, query,
8746f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                        '[', ']', "...", 5);
8756f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro            }
8766f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro        }
8776f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro        return value;
8786f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro    }
8796f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro
8802f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa    protected Long getStoredLongValue(Uri uri, String selection, String[] selectionArgs,
8812f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa            String column) {
8822f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa        Long value = null;
8832f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa        Cursor c = mResolver.query(uri, new String[] { column }, selection, selectionArgs, null);
8842f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa        try {
8852f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa            assertEquals("Record count", 1, c.getCount());
8862f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa
8872f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa            if (c.moveToFirst()) {
8882f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa                value = c.getLong(c.getColumnIndex(column));
8892f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa            }
8902f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa        } finally {
8912f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa            c.close();
8922f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa        }
8932f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa        return value;
8942f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa    }
8952f830d3bb66f780937203e9738e046841a070e73Daisuke Miyakawa
896f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro    protected Long getStoredLongValue(Uri uri, String column) {
897f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro        return getStoredLongValue(uri, null, null, column);
898f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro    }
899f547fd54d7933e1c03af4a8dc10560c71c38f6b8Dave Santoro
90020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertStoredValues(Uri rowUri, ContentValues expectedValues) {
901bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov        assertStoredValues(rowUri, null, null, expectedValues);
902bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov    }
903bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov
904892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    protected void assertStoredValues(Uri rowUri, ContentValues[] expectedValues) {
905892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        assertStoredValues(rowUri, null, null, expectedValues);
906892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    }
907892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov
908bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov    protected void assertStoredValues(Uri rowUri, String selection, String[] selectionArgs,
909bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov            ContentValues expectedValues) {
910bce6ee29f2d971ceae2bfce85a06bb3ecec6537aDmitri Plotnikov        Cursor c = mResolver.query(rowUri, null, selection, selectionArgs, null);
91120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
91220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
91320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.moveToFirst();
91420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertCursorValues(c, expectedValues);
91520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
91620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
91720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
91820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
91920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
920ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    protected void assertStoredValuesWithProjection(Uri rowUri, ContentValues expectedValues) {
92123ba865a6d204ba4aa29d2fad9989e9c44351e81Makoto Onuki        assertStoredValuesWithProjection(rowUri, new ContentValues[] {expectedValues});
92223ba865a6d204ba4aa29d2fad9989e9c44351e81Makoto Onuki    }
92323ba865a6d204ba4aa29d2fad9989e9c44351e81Makoto Onuki
92423ba865a6d204ba4aa29d2fad9989e9c44351e81Makoto Onuki    protected void assertStoredValuesWithProjection(Uri rowUri, ContentValues[] expectedValues) {
92523ba865a6d204ba4aa29d2fad9989e9c44351e81Makoto Onuki        assertTrue("Need at least one ContentValues for this test", expectedValues.length > 0);
92623ba865a6d204ba4aa29d2fad9989e9c44351e81Makoto Onuki        Cursor c = mResolver.query(rowUri, buildProjection(expectedValues[0]), null, null, null);
927ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        try {
92823ba865a6d204ba4aa29d2fad9989e9c44351e81Makoto Onuki            assertEquals("Record count", expectedValues.length, c.getCount());
929ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            c.moveToFirst();
930ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            assertCursorValues(c, expectedValues);
931ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        } finally {
932ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            c.close();
933ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        }
934ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
935ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
936892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    protected void assertStoredValues(
937892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            Uri rowUri, String selection, String[] selectionArgs, ContentValues[] expectedValues) {
938892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        Cursor c = mResolver.query(rowUri, null, selection, selectionArgs, null);
939892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        try {
940892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            assertEquals("Record count", expectedValues.length, c.getCount());
941892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            assertCursorValues(c, expectedValues);
942892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        } finally {
943892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            c.close();
944892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        }
945892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    }
946892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov
9477d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa    protected void assertStoredValuesOrderly(Uri rowUri, ContentValues[] expectedValues) {
9487d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        assertStoredValuesOrderly(rowUri, null, null, expectedValues);
9497d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa    }
9507d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa
9517d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa    protected void assertStoredValuesOrderly(Uri rowUri, String selection,
9527d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa            String[] selectionArgs, ContentValues[] expectedValues) {
9537d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        Cursor c = mResolver.query(rowUri, null, selection, selectionArgs, null);
9547d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        try {
9557d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa            assertEquals("Record count", expectedValues.length, c.getCount());
9567d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa            assertCursorValuesOrderly(c, expectedValues);
9577d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        } finally {
9587d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa            c.close();
9597d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        }
9607d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa    }
9617d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa
96220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    /**
96320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * Constructs a selection (where clause) out of all supplied values, uses it
96420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * to query the provider and verifies that a single row is returned and it
96520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     * has the same values as requested.
96620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov     */
96720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertSelection(Uri uri, ContentValues values, String idColumn, long id) {
968ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        assertSelection(uri, values, idColumn, id, null);
969ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
970ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
971ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    public void assertSelectionWithProjection(Uri uri, ContentValues values, String idColumn,
972ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            long id) {
973ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        assertSelection(uri, values, idColumn, id, buildProjection(values));
974ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
975ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
976ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    private void assertSelection(Uri uri, ContentValues values, String idColumn, long id,
977ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            String[] projection) {
97820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        StringBuilder sb = new StringBuilder();
97920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        ArrayList<String> selectionArgs = new ArrayList<String>(values.size());
980c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        if (idColumn != null) {
981c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            sb.append(idColumn).append("=").append(id);
982c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov        }
98320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        Set<Map.Entry<String, Object>> entries = values.valueSet();
98420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        for (Map.Entry<String, Object> entry : entries) {
98520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String column = entry.getKey();
98620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            Object value = entry.getValue();
987c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            if (sb.length() != 0) {
988c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov                sb.append(" AND ");
989c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            }
990c70dc0e38ff82c6e6d6b7458637c54fbdf446aacDmitri Plotnikov            sb.append(column);
99120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            if (value == null) {
99220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                sb.append(" IS NULL");
99320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            } else {
99420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                sb.append("=?");
99520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                selectionArgs.add(String.valueOf(value));
99620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            }
99720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
99820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
999ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Cursor c = mResolver.query(uri, projection, sb.toString(), selectionArgs.toArray(new String[0]),
100020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                null);
100120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        try {
100220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertEquals("Record count", 1, c.getCount());
100320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.moveToFirst();
100420a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            assertCursorValues(c, values);
100520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        } finally {
100620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            c.close();
100720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
100820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
100920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov
1010c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov    protected void assertCursorValue(Cursor cursor, String column, Object expectedValue) {
1011c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov        String actualValue = cursor.getString(cursor.getColumnIndex(column));
1012c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov        assertEquals("Column " + column, String.valueOf(expectedValue),
1013c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov                String.valueOf(actualValue));
1014c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov    }
1015c918b0d3ab17a45a392748f43956b927c83eb402Dmitri Plotnikov
101620a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    protected void assertCursorValues(Cursor cursor, ContentValues expectedValues) {
1017892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        StringBuilder message = new StringBuilder();
1018892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        boolean result = equalsWithExpectedValues(cursor, expectedValues, message);
1019892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        assertTrue(message.toString(), result);
1020892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    }
1021892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov
1022892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    protected void assertCursorValues(Cursor cursor, ContentValues[] expectedValues) {
1023892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        StringBuilder message = new StringBuilder();
1024892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        for (ContentValues v : expectedValues) {
1025892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            boolean found = false;
1026892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            cursor.moveToPosition(-1);
1027892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            while (cursor.moveToNext()) {
1028892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                found = equalsWithExpectedValues(cursor, v, message);
1029892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                if (found) {
1030892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                    break;
1031892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                }
1032892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            }
1033892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            assertTrue("Expected values can not be found " + v + message.toString(), found);
1034892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        }
1035892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    }
1036892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov
10377d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa    protected void assertCursorValuesOrderly(Cursor cursor, ContentValues[] expectedValues) {
10387d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        StringBuilder message = new StringBuilder();
10397d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        cursor.moveToPosition(-1);
10407d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        for (ContentValues v : expectedValues) {
10417d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa            assertTrue(cursor.moveToNext());
10427d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa            assertTrue("ContentValues didn't match " + v + message.toString(),
10437d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa                    equalsWithExpectedValues(cursor, v, message));
10447d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa        }
10457d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa    }
10467d82ae92714f2132e3a0971d844ae8cdf10d76e7Daisuke Miyakawa
1047892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov    private boolean equalsWithExpectedValues(Cursor cursor, ContentValues expectedValues,
1048892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            StringBuilder msgBuffer) {
104920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        Set<Map.Entry<String, Object>> entries = expectedValues.valueSet();
105020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        for (Map.Entry<String, Object> entry : entries) {
105120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String column = entry.getKey();
105220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            int index = cursor.getColumnIndex(column);
1053892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            if (index == -1) {
1054892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                msgBuffer.append("No such column: ").append(column);
1055892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                return false;
1056892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            }
105720a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            Object expectedValue = expectedValues.get(column);
105820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            String value;
105920a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            if (expectedValue instanceof byte[]) {
106020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                expectedValue = Hex.encodeHex((byte[])expectedValue, false);
106120a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                value = Hex.encodeHex(cursor.getBlob(index), false);
106220a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            } else {
106320a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov                expectedValue = expectedValues.getAsString(column);
10646f983fd835f0cdd5ac7931ccd49d44e9ea4c87c0Dave Santoro                value = getCursorStringValue(cursor, column);
106520a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov            }
1066892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            if (expectedValue != null && !expectedValue.equals(value) || value != null
1067892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                    && !value.equals(expectedValue)) {
1068892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                msgBuffer
1069892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append("Column value ")
1070892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(column)
1071892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(" expected <")
1072892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(expectedValue)
1073892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(">, but was <")
1074892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append(value)
1075892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                        .append('>');
1076892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov                return false;
1077892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov            }
107820a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov        }
1079892a3d9ded5c64a63ae3d5d5c52c59528b466c93Dmitri Plotnikov        return true;
108020a94c86ede7380c8dd8df2f6a72b3c00ac1bed8Dmitri Plotnikov    }
108133b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov
1082ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    private String[] buildProjection(ContentValues values) {
1083ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        String[] projection = new String[values.size()];
1084ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        Iterator<Entry<String, Object>> iter = values.valueSet().iterator();
1085ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        for (int i = 0; i < projection.length; i++) {
1086ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov            projection[i] = iter.next().getKey();
1087ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        }
1088ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov        return projection;
1089ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov    }
1090ff2de103f7e3eeeff4665ef63f07460fef053d6dDmitri Plotnikov
109133b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov    protected int getCount(Uri uri, String selection, String[] selectionArgs) {
109233b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        Cursor c = mResolver.query(uri, null, selection, selectionArgs, null);
109333b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        try {
109433b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov            return c.getCount();
109533b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        } finally {
109633b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov            c.close();
109733b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov        }
109833b41fdb8d7c3c654cb070799c9d6e2b4ab16078Dmitri Plotnikov    }
1099a908fb5f39aa2021662a6cc317cc7e4db2d8bfb0Dmitri Plotnikov
1100b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    public static void dump(ContentResolver resolver, boolean aggregatedOnly) {
1101b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
1102b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts._ID,
1103b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts.DISPLAY_NAME
1104b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
1105b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String selection = null;
1106b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        if (aggregatedOnly) {
1107b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            selection = Contacts._ID
1108b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                    + " IN (SELECT contact_id" +
1109b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                    		" FROM raw_contacts GROUP BY contact_id HAVING count(*) > 1)";
1110b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
1111b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
1112b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(Contacts.CONTENT_URI, projection, selection, null,
1113b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Contacts.DISPLAY_NAME);
1114b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
1115b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            long contactId = c.getLong(0);
1116b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("Contact   ", String.format("%5d %s", contactId, c.getString(1)));
1117b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            dumpRawContacts(resolver, contactId);
1118b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("          ", ".");
1119b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
1120b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
1121b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
1122b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
1123b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    private static void dumpRawContacts(ContentResolver resolver, long contactId) {
1124b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
1125b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                RawContacts._ID,
1126b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
1127b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(RawContacts.CONTENT_URI, projection, RawContacts.CONTACT_ID + "="
1128b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                + contactId, null, null);
1129b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
1130b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            long rawContactId = c.getLong(0);
1131b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            Log.i("RawContact", String.format("      %-5d", rawContactId));
1132b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            dumpData(resolver, rawContactId);
1133b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
1134b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
1135b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
1136b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov
1137b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    private static void dumpData(ContentResolver resolver, long rawContactId) {
1138b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        String[] projection = new String[] {
1139b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.MIMETYPE,
1140b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA1,
1141b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA2,
1142b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Data.DATA3,
1143b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        };
1144b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        Cursor c = resolver.query(Data.CONTENT_URI, projection, Data.RAW_CONTACT_ID + "="
1145b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                + rawContactId, null, Data.MIMETYPE);
1146b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        while(c.moveToNext()) {
1147b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            String mimetype = c.getString(0);
1148b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            if (Photo.CONTENT_ITEM_TYPE.equals(mimetype)) {
1149b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Log.i("Photo     ", "");
1150b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            } else {
1151b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                mimetype = mimetype.substring(mimetype.indexOf('/') + 1);
1152b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                Log.i("Data      ", String.format("            %-10s %s,%s,%s", mimetype,
1153b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov                        c.getString(1), c.getString(2), c.getString(3)));
1154b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov            }
1155b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        }
1156b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov        c.close();
1157b58058243bcd4405ad5ba5c3a465c46a35182ff7Dmitri Plotnikov    }
1158285b771bc955305fa6d49ca23f808cecc8a13d5eDmitri Plotnikov
115981d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov    protected void assertNetworkNotified(boolean expected) {
11609a6be1610fdf40c2f7f04cfe4b66fde3a35940dcDmitri Plotnikov        assertEquals(expected, (getContactsProvider()).isNetworkNotified());
116181d6a78dffd57f24f9aaecb6cd54e4084c3c9846Dmitri Plotnikov    }
11625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
116309ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov    protected void assertProjection(Uri uri, String[] expectedProjection) {
116409ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov        Cursor cursor = mResolver.query(uri, null, "0", null, null);
116509ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov        String[] actualProjection = cursor.getColumnNames();
116609ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov        MoreAsserts.assertEquals("Incorrect projection for URI: " + uri,
116709ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov                Sets.newHashSet(expectedProjection), Sets.newHashSet(actualProjection));
116809ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov        cursor.close();
116909ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov    }
117009ae48b82b17e24016b14a1ab64706222ab1071fDmitri Plotnikov
11715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    /**
11725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * A contact in the database, and the attributes used to create it.  Construct using
11735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * {@link GoldenContactBuilder#build()}.
11745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     */
11755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    public final class GoldenContact {
11765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final long rawContactId;
11785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final long contactId;
11805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String givenName;
11825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String familyName;
11845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String nickname;
11865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final byte[] photo;
11885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String company;
11905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String title;
11925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String phone;
11945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private final String email;
11965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private GoldenContact(GoldenContactBuilder builder, long rawContactId, long contactId) {
11985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
11995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            this.rawContactId = rawContactId;
12005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            this.contactId = contactId;
12015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            givenName = builder.givenName;
12025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            familyName = builder.familyName;
12035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            nickname = builder.nickname;
12045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            photo = builder.photo;
12055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            company = builder.company;
12065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            title = builder.title;
12075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            phone = builder.phone;
12085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            email = builder.email;
12095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12115b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public void delete() {
12125b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            Uri rawContactUri = ContentUris.withAppendedId(RawContacts.CONTENT_URI, rawContactId);
12135b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.delete(rawContactUri, null, null);
12145b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12155b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12165b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12175b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the index of the contact in table "raw_contacts"
12185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public long getRawContactId() {
12205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return rawContactId;
12215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the index of the contact in table "contacts"
12255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public long getContactId() {
12275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return contactId;
12285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12312d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill         * Returns the lookup key for the contact.
12322d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill         */
12332d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        public String getLookupKey() {
12342d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill            return queryLookupKey(contactId);
12352d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        }
12362d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill
12372d2ec88b7af615b2f05e987da45425be9cace1baTom O'Neill        /**
12385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's given name.
12395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getGivenName() {
12415b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return givenName;
12425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12435b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12445b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12455b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's family name.
12465b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12475b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getFamilyName() {
12485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return familyName;
12495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's nickname.
12535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getNickname() {
12555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return nickname;
12565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Return's the contact's photo
12605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public byte[] getPhoto() {
12625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return photo;
12635b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Return's the company at which the contact works.
12675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getCompany() {
12695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return company;
12705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's job title.
12745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getTitle() {
12765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return title;
12775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's phone number
12815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getPhone() {
12835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return phone;
12845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
12875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Returns the contact's email address
12885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
12895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public String getEmail() {
12905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return email;
12915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
12925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     }
12935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
12945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    /**
12955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * Builds {@link GoldenContact} objects.  Unspecified boolean objects default to false.
12965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     * Unspecified String objects default to null.
12975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill     */
12985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    public final class GoldenContactBuilder {
12995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String givenName;
13015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String familyName;
13035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String nickname;
13055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private byte[] photo;
13075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String company;
13095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String title;
13115b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13125b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String phone;
13135b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13145b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private String email;
13155b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13165b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13175b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's given and family names.
13185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         *
13195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * TODO(dplotnikov): inline, or should we require them to set both names if they set either?
13205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder name(String givenName, String familyName) {
13225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return givenName(givenName).familyName(familyName);
13235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's given name.
13275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder givenName(String value) {
13295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            givenName = value;
13305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
13315b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13325b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13335b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13345b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's family name.
13355b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13365b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder familyName(String value) {
13375b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            familyName = value;
13385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
13395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13415b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's nickname.
13435b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13445b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder nickname(String value) {
13455b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            nickname = value;
13465b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
13475b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's photo.
13515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder photo(byte[] value) {
13535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            photo = value;
13545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
13555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The company at which the contact works.
13595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder company(String value) {
13615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            company = value;
13625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
13635b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's job title.
13675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder title(String value) {
13695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            title = value;
13705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
13715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's phone number.
13755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder phone(String value) {
13775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            phone = value;
13785b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
13795b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13805b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13815b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13825b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * The contact's email address; also sets their IM status to {@link StatusUpdates#OFFLINE}
13835b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * with a presence of "Coding for Android".
13845b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13855b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContactBuilder email(String value) {
13865b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            email = value;
13875b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return this;
13885b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
13895b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13905b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        /**
13915b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         * Builds the {@link GoldenContact} specified by this builder.
13925b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill         */
13935b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        public GoldenContact build() {
13945b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13955b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            final long groupId = createGroup(mAccount, "gsid1", "title1");
13965b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
13975b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            long rawContactId = createRawContact();
13985b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            insertGroupMembership(rawContactId, groupId);
13995b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14005b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (givenName != null || familyName != null) {
14015b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertStructuredName(rawContactId, givenName, familyName);
14025b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
14035b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (nickname != null) {
14045b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertNickname(rawContactId, nickname);
14055b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
14065b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (photo != null) {
14075b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertPhoto(rawContactId);
14085b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
14095b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (company != null || title != null) {
14105b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertOrganization(rawContactId);
14115b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
14125b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (email != null) {
14135b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertEmail(rawContactId);
14145b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
14155b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (phone != null) {
14165b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                insertPhone(rawContactId);
14175b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
14185b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14195b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            long contactId = queryContactId(rawContactId);
14205b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14215b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            return new GoldenContact(this, rawContactId, contactId);
14225b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
14235b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14245b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertPhoto(long rawContactId) {
14255b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
14265b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
14275b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Photo.CONTENT_ITEM_TYPE);
14285b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Photo.PHOTO, photo);
14295b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
14305b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
14315b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14325b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertOrganization(long rawContactId) {
14335b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14345b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
14355b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
14365b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Organization.CONTENT_ITEM_TYPE);
14375b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Organization.TYPE, Organization.TYPE_WORK);
14385b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (company != null) {
14395b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                values.put(Organization.COMPANY, company);
14405b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
14415b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            if (title != null) {
14425b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill                values.put(Organization.TITLE, title);
14435b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            }
14445b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
14455b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
14465b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14475b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertEmail(long rawContactId) {
14485b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14495b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
14505b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
14515b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Email.CONTENT_ITEM_TYPE);
14525b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Email.TYPE, Email.TYPE_WORK);
14535b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Email.DATA, "foo@acme.com");
14545b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
14555b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14565b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            int protocol = Im.PROTOCOL_GOOGLE_TALK;
14575b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14585b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.clear();
14595b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PROTOCOL, protocol);
14605b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.IM_HANDLE, email);
14615b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.IM_ACCOUNT, "foo");
14625b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PRESENCE_STATUS, StatusUpdates.OFFLINE);
1463aabcd1d34a71ad06ee0a9395331540484f1ceb17Vasu Nori            values.put(StatusUpdates.CHAT_CAPABILITY, StatusUpdates.CAPABILITY_HAS_CAMERA);
14645b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(StatusUpdates.PRESENCE_CUSTOM_STATUS, "Coding for Android");
14655b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(StatusUpdates.CONTENT_URI, values);
14665b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
14675b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill
14685b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        private void insertPhone(long rawContactId) {
14695b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            ContentValues values = new ContentValues();
14705b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.RAW_CONTACT_ID, rawContactId);
14715b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.MIMETYPE, Phone.CONTENT_ITEM_TYPE);
14725b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Data.IS_PRIMARY, 1);
14735b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Phone.TYPE, Phone.TYPE_HOME);
14745b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            values.put(Phone.NUMBER, phone);
14755b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill            mResolver.insert(Data.CONTENT_URI, values);
14765b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill        }
14775b3b3ec368625c07a2c94d2199a75099d4d9b8a9Tom O'Neill    }
1478d35d9c748af4c3182679c4c546137acfc11eb7a8Dmitri Plotnikov}
1479