ProviderTests.java revision 2f288864b621cfb5aee44eda27df463460d33dd3
17143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler/*
27143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler * Copyright (C) 2009 The Android Open Source Project
37143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler *
417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie * Licensed under the Apache License, Version 2.0 (the "License"); you may not
517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie * use this file except in compliance with the License. You may obtain a copy of
617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie * the License at
77143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler *
817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie * http://www.apache.org/licenses/LICENSE-2.0
97143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler *
107143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler * Unless required by applicable law or agreed to in writing, software
1117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
1217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
1317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie * License for the specific language governing permissions and limitations under
1417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie * the License.
157143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler */
167143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
177143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadlerpackage com.android.email.provider;
187143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
196e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.content.ContentResolver;
206e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.content.ContentUris;
216e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.content.ContentValues;
226e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.content.Context;
2332bed4bb8e23d7322ab338773d135845f392d3cfBen Komaloimport android.content.ContextWrapper;
246e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.database.Cursor;
256e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.database.sqlite.SQLiteDatabase;
266e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.net.Uri;
276e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.os.Bundle;
286e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.os.Environment;
296e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.os.Parcel;
306e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.test.MoreAsserts;
316e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport android.test.ProviderTestCase2;
3252e66110d8934cf0a3dd632101a56d4ef45b549aBen Komaloimport android.test.suitebuilder.annotation.LargeTest;
3352e66110d8934cf0a3dd632101a56d4ef45b549aBen Komaloimport android.test.suitebuilder.annotation.MediumTest;
3452e66110d8934cf0a3dd632101a56d4ef45b549aBen Komaloimport android.test.suitebuilder.annotation.SmallTest;
357037a0bd3d8e925a9115f475f5c0d05ddae2eeeeJerry Xieimport android.test.suitebuilder.annotation.Suppress;
366e418aa41a17136be0dddb816d843428a0a1e722Marc Blank
3732bed4bb8e23d7322ab338773d135845f392d3cfBen Komaloimport com.android.email.provider.EmailProvider.AttachmentService;
38f5418f1f93b02e7fab9f15eb201800b65510998eMarc Blankimport com.android.emailcommon.provider.Account;
39a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent;
40a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent.AccountColumns;
41a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent.Attachment;
42a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent.AttachmentColumns;
43a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent.Body;
44a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent.BodyColumns;
45a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent.MailboxColumns;
46a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent.Message;
47a7bc0319a75184ad706bb35c049af107ac3688e6Marc Blankimport com.android.emailcommon.provider.EmailContent.MessageColumns;
482bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blankimport com.android.emailcommon.provider.EmailContent.PolicyColumns;
4912b82d9374947c9268217f45befe8a74bd9b60d7Ben Komaloimport com.android.emailcommon.provider.HostAuth;
5053ea83ebf91f820692e8fa8e781f5cc982dd94dbBen Komaloimport com.android.emailcommon.provider.Mailbox;
516e418aa41a17136be0dddb816d843428a0a1e722Marc Blankimport com.android.emailcommon.provider.Policy;
5276f614720db6a282c3e890969a600585122cdcc5Marc Blankimport com.android.emailcommon.utility.TextUtilities;
5331d9acbf0623872f9d4a2b3210b5970854b654c7Marc Blankimport com.android.emailcommon.utility.Utility;
547143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
55758a532fce2f672673d38b2daa5f67eb757b118bMarc Blankimport java.io.File;
56758a532fce2f672673d38b2daa5f67eb757b118bMarc Blankimport java.io.IOException;
57758a532fce2f672673d38b2daa5f67eb757b118bMarc Blankimport java.util.ArrayList;
58758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
597143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler/**
607143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler * Tests of the Email provider.
61758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank *
627143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler * You can run this entire test case with:
637143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler *   runtest -c com.android.email.provider.ProviderTests email
64261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki *
65261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki * TODO: Add tests for cursor notification mechanism.  (setNotificationUri and notifyChange)
66261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki * We can't test the entire notification mechanism with a mock content resolver, because which URI
67261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki * to notify when notifyChange() is called is in the actual content resolver.
68261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki * Implementing the same mechanism in a mock one is pointless.  Instead what we could do is check
69261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki * what notification URI each cursor has, and with which URI is notified when
70261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki * inserting/updating/deleting.  (The former require a new method from AbstractCursor)
717143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler */
727037a0bd3d8e925a9115f475f5c0d05ddae2eeeeJerry Xie@Suppress
7352e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo@LargeTest
747143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadlerpublic class ProviderTests extends ProviderTestCase2<EmailProvider> {
75758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
769d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    private EmailProvider mProvider;
779d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    private Context mMockContext;
787143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
797143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public ProviderTests() {
8031d9acbf0623872f9d4a2b3210b5970854b654c7Marc Blank        super(EmailProvider.class, EmailContent.AUTHORITY);
817143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
82758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
8317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    // TODO: move this out to a common place. There are other places that have
8417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    // similar mocks.
8532bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    /**
8632bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo     * Private context wrapper used to add back getPackageName() for these tests.
8732bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo     */
8832bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    private static class MockContext2 extends ContextWrapper {
8932bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
9032bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        private final Context mRealContext;
9132bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
9232bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        public MockContext2(Context mockContext, Context realContext) {
9332bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo            super(mockContext);
9432bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo            mRealContext = realContext;
9532bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        }
9632bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
9732bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        @Override
9832bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        public Context getApplicationContext() {
9932bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo            return this;
10032bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        }
10132bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
10232bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        @Override
10332bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        public String getPackageName() {
10432bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo            return mRealContext.getPackageName();
10532bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        }
10632bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
10732bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        @Override
10832bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        public Object getSystemService(String name) {
10932bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo            return mRealContext.getSystemService(name);
11032bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        }
11132bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    }
11232bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
11332bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    private static final AttachmentService MOCK_ATTACHMENT_SERVICE = new AttachmentService() {
11432bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        @Override
11532bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        public void attachmentChanged(Context context, long id, int flags) {
11632bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo            // Noop. Don't download attachments.
11732bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        }
11832bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    };
11932bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
1207143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    @Override
1217143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public void setUp() throws Exception {
1227143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        super.setUp();
12332bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        mMockContext = new MockContext2(getMockContext(), getContext());
1249d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        mProvider = getProvider();
12532bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        mProvider.injectAttachmentService(MOCK_ATTACHMENT_SERVICE);
126349055aad47184b72cd86de1f37ac1b7557d2e70Marc Blank        // Invalidate all caches, since we reset the database for each test
1276e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        ContentCache.invalidateAllCaches();
1287143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
1297143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
1307143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    @Override
1317143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public void tearDown() throws Exception {
1327143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        super.tearDown();
13332bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        mProvider.injectAttachmentService(null);
1347143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
1357143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
1367143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
1370d00889f83fc8c1d69039a2c849b630fb4dd28ddAndrew Stadler     * TODO: Database upgrade tests
1380d00889f83fc8c1d69039a2c849b630fb4dd28ddAndrew Stadler     */
1390d00889f83fc8c1d69039a2c849b630fb4dd28ddAndrew Stadler
14017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    // ////////////////////////////////////////////////////////
14117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    // //// Utility methods
14217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    // ////////////////////////////////////////////////////////
14322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
14422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    /** Sets the message count of all mailboxes to {@code -1}. */
14522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    private void setMinusOneToMessageCounts() {
14622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        ContentValues values = new ContentValues();
14722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        values.put(MailboxColumns.MESSAGE_COUNT, -1);
14822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
14917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // EmailProvider.update() doesn't allow updating messageCount, so
15017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // directly use the DB.
15122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        SQLiteDatabase db = getProvider().getDatabase(mMockContext);
15222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        db.update(Mailbox.TABLE_NAME, values, null, null);
15322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    }
15422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
15522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    /** Returns the number of messages in a mailbox. */
15622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    private int getMessageCount(long mailboxId) {
15722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        return Utility.getFirstRowInt(mMockContext,
15822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy                ContentUris.withAppendedId(Mailbox.CONTENT_URI, mailboxId),
15917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                new String[] {MailboxColumns.MESSAGE_COUNT},
16017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                null,
16117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                null,
16217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                null,
16317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                0);
16422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    }
16522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
16622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    /** Creates a new message. */
16717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    private static Message createMessage(
16817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            Context c, Mailbox b, boolean starred, boolean read, int flagLoaded) {
16917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message = ProviderTestUtils.setupMessage("1",
17017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                b.mAccountKey,
17117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                b.mId,
17217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
17317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
17417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                c,
17517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                starred,
17617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                read);
17722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        message.mFlagLoaded = flagLoaded;
17822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        message.save(c);
17922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        return message;
18022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    }
18122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
18217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    // ////////////////////////////////////////////////////////
18317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    // //// The tests
18417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    // ////////////////////////////////////////////////////////
18522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
1860d00889f83fc8c1d69039a2c849b630fb4dd28ddAndrew Stadler    /**
1877143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * Test simple account save/retrieve
1887143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
18952e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo    @SmallTest
1907143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public void testAccountSave() {
19128448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Account account1 = ProviderTestUtils.setupAccount("account-save", true, mMockContext);
1927143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long account1Id = account1.mId;
193758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
194f5418f1f93b02e7fab9f15eb201800b65510998eMarc Blank        Account account2 = Account.restoreAccountWithId(mMockContext, account1Id);
195758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
19628448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        ProviderTestUtils.assertAccountEqual("testAccountSave", account1, account2);
1977143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
19841f775bbc731b25854da9f5a80c66a4aa1eb0af9satok
1994a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler    /**
200da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler     * Test simple account save/retrieve with predefined hostauth records
201da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler     */
20252e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo    @SmallTest
203da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler    public void testAccountSaveHostAuth() {
204da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler        Account account1 = ProviderTestUtils.setupAccount("account-hostauth", false, mMockContext);
205da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler        // add hostauth data, which should be saved the first time
20617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        account1.mHostAuthRecv =
20717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupHostAuth("account-hostauth-recv", -1, false, mMockContext);
20817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        account1.mHostAuthSend =
20917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupHostAuth("account-hostauth-send", -1, false, mMockContext);
210da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler        account1.save(mMockContext);
211da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler        long account1Id = account1.mId;
212da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler
213da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler        // Confirm account reads back correctly
214f5418f1f93b02e7fab9f15eb201800b65510998eMarc Blank        Account account1get = Account.restoreAccountWithId(mMockContext, account1Id);
215da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler        ProviderTestUtils.assertAccountEqual("testAccountSave", account1, account1get);
216da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler
217da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler        // Confirm hostauth fields can be accessed & read back correctly
21817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        HostAuth hostAuth1get =
21917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                HostAuth.restoreHostAuthWithId(mMockContext, account1get.mHostAuthKeyRecv);
22017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        ProviderTestUtils.assertHostAuthEqual(
22117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                "testAccountSaveHostAuth-recv", account1.mHostAuthRecv, hostAuth1get);
22217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        HostAuth hostAuth2get =
22317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                HostAuth.restoreHostAuthWithId(mMockContext, account1get.mHostAuthKeySend);
22417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        ProviderTestUtils.assertHostAuthEqual(
22517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                "testAccountSaveHostAuth-send", account1.mHostAuthSend, hostAuth2get);
226da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler    }
227da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler
228040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy    public void testAccountGetHostAuthSend() {
229040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        Account account = ProviderTestUtils.setupAccount("account-hostauth", false, mMockContext);
23017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        account.mHostAuthSend =
23117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupHostAuth("account-hostauth-send", -1, false, mMockContext);
232040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        account.save(mMockContext);
233040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        HostAuth authGet;
234040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        HostAuth authTest;
235040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
236040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        authTest = account.mHostAuthSend;
237040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(authTest);
238040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertTrue(account.mHostAuthKeySend != 0);
239040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
240040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        // HostAuth is not changed
241040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        authGet = account.getOrCreateHostAuthSend(mMockContext);
242040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertTrue(authGet == authTest); // return the same object
243040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
244040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        // New HostAuth; based upon mHostAuthKeyRecv
24517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        authTest = HostAuth.restoreHostAuthWithId(mMockContext, account.mHostAuthKeySend);
246040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        account.mHostAuthSend = null;
247040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        authGet = account.getOrCreateHostAuthSend(mMockContext);
248040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(authGet);
249040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(account.mHostAuthSend);
250040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        ProviderTestUtils.assertHostAuthEqual("testAccountGetHostAuthSend-1", authTest, authGet);
251040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
252040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        // New HostAuth; completely empty
25312b82d9374947c9268217f45befe8a74bd9b60d7Ben Komalo        authTest = new HostAuth();
254040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        account.mHostAuthSend = null;
255040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        account.mHostAuthKeySend = 0;
256040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        authGet = account.getOrCreateHostAuthSend(mMockContext);
257040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(authGet);
258040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(account.mHostAuthSend);
259040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        ProviderTestUtils.assertHostAuthEqual("testAccountGetHostAuthSendv-2", authTest, authGet);
260040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy    }
261040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
262040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy    public void testAccountGetHostAuthRecv() {
263040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        Account account = ProviderTestUtils.setupAccount("account-hostauth", false, mMockContext);
26417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        account.mHostAuthRecv =
26517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupHostAuth("account-hostauth-recv", -1, false, mMockContext);
266040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        account.save(mMockContext);
267040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        HostAuth authGet;
268040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        HostAuth authTest;
269040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
270040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        authTest = account.mHostAuthRecv;
271040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(authTest);
272040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertTrue(account.mHostAuthKeyRecv != 0);
273040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
274040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        // HostAuth is not changed
275040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        authGet = account.getOrCreateHostAuthRecv(mMockContext);
276040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertTrue(authGet == authTest); // return the same object
277040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
278040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        // New HostAuth; based upon mHostAuthKeyRecv
27917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        authTest = HostAuth.restoreHostAuthWithId(mMockContext, account.mHostAuthKeyRecv);
280040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        account.mHostAuthRecv = null;
281040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        authGet = account.getOrCreateHostAuthRecv(mMockContext);
282040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(authGet);
283040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(account.mHostAuthRecv);
284040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        ProviderTestUtils.assertHostAuthEqual("testAccountGetHostAuthRecv-1", authTest, authGet);
285040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
286040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        // New HostAuth; completely empty
28712b82d9374947c9268217f45befe8a74bd9b60d7Ben Komalo        authTest = new HostAuth();
288040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        account.mHostAuthRecv = null;
289040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        account.mHostAuthKeyRecv = 0;
290040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        authGet = account.getOrCreateHostAuthRecv(mMockContext);
291040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(authGet);
292040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        assertNotNull(account.mHostAuthRecv);
293040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy        ProviderTestUtils.assertHostAuthEqual("testAccountGetHostAuthRecv-2", authTest, authGet);
294040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy    }
295040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy
296da8836a76cd8a6eaa7e3693eeacc6393870b2066Andrew Stadler    /**
297040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy     * Simple test of account parceling.  The rather torturous path is to ensure that the
2984a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler     * account is really flattened all the way down to a parcel and back.
2994a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler     */
3004a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler    public void testAccountParcel() {
3014a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        Account account1 = ProviderTestUtils.setupAccount("parcel", false, mMockContext);
3024a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        Bundle b = new Bundle();
3034a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        b.putParcelable("account", account1);
3044a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        Parcel p = Parcel.obtain();
3054a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        b.writeToParcel(p, 0);
30617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        p.setDataPosition(0); // rewind it for reading
3074a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        Bundle b2 = new Bundle(Account.class.getClassLoader());
3084a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        b2.readFromParcel(p);
3094a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        Account account2 = (Account) b2.getParcelable("account");
3104a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        p.recycle();
3114a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler
3124a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler        ProviderTestUtils.assertAccountEqual("testAccountParcel", account1, account2);
3134a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler    }
3144a8c70c09be3914ded18031b4cca5a6d867de0d3Andrew Stadler
315e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki    /**
316e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki     * Test for {@link Account#getShortcutSafeUri()} and
317040ddf60cfef4aaecf4bfe1f897fce3248d777a4Todd Kennedy     * {@link Account#getAccountIdFromShortcutSafeUri}.
318e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki     */
319e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki    public void testAccountShortcutSafeUri() {
320e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        final Account account1 = ProviderTestUtils.setupAccount("account-1", true, mMockContext);
321e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        final Account account2 = ProviderTestUtils.setupAccount("account-2", true, mMockContext);
322e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        final long account1Id = account1.mId;
323e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        final long account2Id = account2.mId;
324e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki
325e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        final Uri uri1 = account1.getShortcutSafeUri();
326e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        final Uri uri2 = account2.getShortcutSafeUri();
327e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki
328e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        // Check the path part of the URIs.
329e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        MoreAsserts.assertEquals(new String[] {"account", account1.mCompatibilityUuid},
330e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki                uri1.getPathSegments().toArray());
331e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        MoreAsserts.assertEquals(new String[] {"account", account2.mCompatibilityUuid},
332e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki                uri2.getPathSegments().toArray());
333e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki
334e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        assertEquals(account1Id, Account.getAccountIdFromShortcutSafeUri(mMockContext, uri1));
335e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        assertEquals(account2Id, Account.getAccountIdFromShortcutSafeUri(mMockContext, uri2));
336e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki
337e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        // Test for the Eclair(2.0-2.1) style URI.
33817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(account1Id, Account.getAccountIdFromShortcutSafeUri(
33917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext, getEclairStyleShortcutUri(account1)));
34017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(account2Id, Account.getAccountIdFromShortcutSafeUri(
34117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext, getEclairStyleShortcutUri(account2)));
342e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki    }
343e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki
344e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki    private static Uri getEclairStyleShortcutUri(Account account) {
345e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        // We used _id instead of UUID only on Eclair(2.0-2.1).
346e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki        return Account.CONTENT_URI.buildUpon().appendEncodedPath("" + account.mId).build();
347e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki    }
348e37881aac562c066e6e666349749e40b7c116a21Makoto Onuki
3498d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank    public void testGetProtocol() {
3508d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        Account account1 = ProviderTestUtils.setupAccount("account-hostauth", false, mMockContext);
3518d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        // add hostauth data, with protocol
35217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        account1.mHostAuthRecv = ProviderTestUtils.setupHostAuth(
35317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                "eas", "account-hostauth-recv", false, mMockContext);
35417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Note that getProtocol uses the receive host auth, so the protocol
35517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // here shouldn't matter
3568d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        // to the test result
35717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        account1.mHostAuthSend = ProviderTestUtils.setupHostAuth(
35817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                "foo", "account-hostauth-send", false, mMockContext);
3598d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        account1.save(mMockContext);
36025144e2b7990c50bb3650faf1aec341837c05348Makoto Onuki        assertEquals("eas", Account.getProtocol(mMockContext, account1.mId));
36125144e2b7990c50bb3650faf1aec341837c05348Makoto Onuki        assertEquals("eas", account1.getProtocol(mMockContext));
36217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Account account2 =
36317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupAccount("account-nohostauth", false, mMockContext);
3648d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        account2.save(mMockContext);
3658d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        // Make sure that we return null when there's no host auth
3668d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        assertNull(Account.getProtocol(mMockContext, account2.mId));
3678d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        assertNull(account2.getProtocol(mMockContext));
3688d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        // And when there's no account
3698d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank        assertNull(Account.getProtocol(mMockContext, 0));
3708d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank    }
3718d8f86e899165772a7d91250b98dfc3c0d78b538Marc Blank
372652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki    public void testAccountIsValidId() {
373652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki        final Account account1 = ProviderTestUtils.setupAccount("account-1", true, mMockContext);
374652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki        final Account account2 = ProviderTestUtils.setupAccount("account-2", true, mMockContext);
375652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki
376652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki        assertTrue(Account.isValidId(mMockContext, account1.mId));
377652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki        assertTrue(Account.isValidId(mMockContext, account2.mId));
378652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki
379652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki        assertFalse(Account.isValidId(mMockContext, 1234567)); // Some random ID
380c133e6f1858e33fabaa6ffa173c1402bf9a98e31Makoto Onuki        assertFalse(Account.isValidId(mMockContext, -1));
381c133e6f1858e33fabaa6ffa173c1402bf9a98e31Makoto Onuki        assertFalse(Account.isValidId(mMockContext, -500));
382652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki    }
383652be6fb3d04a4ceba6b765cd3160cdaef9e6107Makoto Onuki
38417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    private final static String[] MAILBOX_UNREAD_COUNT_PROJECTION =
38517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            new String[] {MailboxColumns.UNREAD_COUNT};
38641f775bbc731b25854da9f5a80c66a4aa1eb0af9satok    private final static int MAILBOX_UNREAD_COUNT_COLMUN = 0;
38741f775bbc731b25854da9f5a80c66a4aa1eb0af9satok
38841f775bbc731b25854da9f5a80c66a4aa1eb0af9satok    /**
38941f775bbc731b25854da9f5a80c66a4aa1eb0af9satok     * Get the value of the unread count in the mailbox of the account.
39041f775bbc731b25854da9f5a80c66a4aa1eb0af9satok     * This can be different from the actual number of unread messages in that mailbox.
39141f775bbc731b25854da9f5a80c66a4aa1eb0af9satok     */
39241f775bbc731b25854da9f5a80c66a4aa1eb0af9satok    private int getUnreadCount(long mailboxId) {
39341f775bbc731b25854da9f5a80c66a4aa1eb0af9satok        String text = null;
39441f775bbc731b25854da9f5a80c66a4aa1eb0af9satok        Cursor c = null;
39541f775bbc731b25854da9f5a80c66a4aa1eb0af9satok        try {
39617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            c = mMockContext.getContentResolver().query(Mailbox.CONTENT_URI,
39717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    MAILBOX_UNREAD_COUNT_PROJECTION, EmailContent.RECORD_ID + "=?",
39817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    new String[] {String.valueOf(mailboxId)}, null);
39941f775bbc731b25854da9f5a80c66a4aa1eb0af9satok            c.moveToFirst();
40041f775bbc731b25854da9f5a80c66a4aa1eb0af9satok            text = c.getString(MAILBOX_UNREAD_COUNT_COLMUN);
40141f775bbc731b25854da9f5a80c66a4aa1eb0af9satok        } finally {
40241f775bbc731b25854da9f5a80c66a4aa1eb0af9satok            c.close();
40341f775bbc731b25854da9f5a80c66a4aa1eb0af9satok        }
40441f775bbc731b25854da9f5a80c66a4aa1eb0af9satok        return Integer.valueOf(text);
40541f775bbc731b25854da9f5a80c66a4aa1eb0af9satok    }
40641f775bbc731b25854da9f5a80c66a4aa1eb0af9satok
4074119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler    private static String[] expectedAttachmentNames =
40817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            new String[] {"attachment1.doc", "attachment2.xls", "attachment3"};
409976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank    // The lengths need to be kept in ascending order
4104119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler    private static long[] expectedAttachmentSizes = new long[] {31415L, 97701L, 151213L};
411976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank
412fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    /*
413fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * Returns null if the message has no body.
414fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     */
415fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    private Body loadBodyForMessageId(long messageId) {
416fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        Cursor c = null;
417fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        try {
41817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            c = mMockContext.getContentResolver().query(EmailContent.Body.CONTENT_URI,
4193dd85723a1af5537e23e4b05bdc361cce9cd42beTony Mantler                    EmailContent.Body.CONTENT_PROJECTION, BodyColumns.MESSAGE_KEY + "=?",
42017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    new String[] {String.valueOf(messageId)}, null);
421fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda            int numBodies = c.getCount();
422fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda            assertTrue("at most one body", numBodies < 2);
4232f288864b621cfb5aee44eda27df463460d33dd3Tony Mantler            return c.moveToFirst() ? EmailContent.getContent(mMockContext, c, Body.class) : null;
424fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        } finally {
425fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda            c.close();
426fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        }
427fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    }
428fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
4297143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
4307143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * Test simple message save/retrieve
431758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank     *
4327143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: serverId vs. serverIntId
4337143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
43452e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo    @MediumTest
4357143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public void testMessageSave() {
43628448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Account account1 = ProviderTestUtils.setupAccount("message-save", true, mMockContext);
4377143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long account1Id = account1.mId;
43828448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
4397143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long box1Id = box1.mId;
4407143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
4417143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // Test a simple message (saved with no body)
44217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
44317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
44417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
44517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
44617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
44717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
4487143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long message1Id = message1.mId;
4497143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        Message message1get = EmailContent.Message.restoreMessageWithId(mMockContext, message1Id);
45028448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        ProviderTestUtils.assertMessageEqual("testMessageSave", message1, message1get);
4517143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
4527143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // Test a message saved with a body
45317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Note that it will read back w/o the text & html so we must extract
45417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // those
45517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message1",
45617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
45717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
45817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
45917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
46017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
4617143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long message2Id = message2.mId;
4627143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        String text2 = message2.mText;
4637143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        String html2 = message2.mHtml;
4646c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        long sourceKey2 = message2.mSourceKey;
4657143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        message2.mText = null;
4667143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        message2.mHtml = null;
467e25698319325d5e9619bc45ae85d861b0f1c5954Marc Blank        message2.mSourceKey = 0;
4687143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        Message message2get = EmailContent.Message.restoreMessageWithId(mMockContext, message2Id);
46928448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        ProviderTestUtils.assertMessageEqual("testMessageSave", message2, message2get);
470758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
4717143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // Now see if there's a body saved with the right stuff
472fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        Body body2 = loadBodyForMessageId(message2Id);
473fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        assertEquals("body text", text2, body2.mTextContent);
474fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        assertEquals("body html", html2, body2.mHtmlContent);
4756c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        assertEquals("source key", sourceKey2, body2.mSourceKey);
47632bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    }
47732bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
47832bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    @MediumTest
47932bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    public void testMessageWithAttachment() {
48032bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        Account account1 = ProviderTestUtils.setupAccount("message-save", true, mMockContext);
48132bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        long account1Id = account1.mId;
48232bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
48332bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        long box1Id = box1.mId;
484976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank
4854119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        // Message with attachments and body
48617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message3 = ProviderTestUtils.setupMessage("message3",
48717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
48817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
48917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
49017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
49117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
492976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        ArrayList<Attachment> atts = new ArrayList<Attachment>();
493976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        for (int i = 0; i < 3; i++) {
4944119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            atts.add(ProviderTestUtils.setupAttachment(
49517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    -1, expectedAttachmentNames[i], expectedAttachmentSizes[i], false,
49617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    mMockContext));
497976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        }
498976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        message3.mAttachments = atts;
4999e2c6bd5f21f2d19eef7ebfe30e6fdf94ede0857Andrew Stadler        message3.save(mMockContext);
500976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        long message3Id = message3.mId;
501976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank
50217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Now check the attachments; there should be three and they should
50317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // match name and size
504fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        Cursor c = null;
505976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        try {
50617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // Note that there is NO guarantee of the order of returned records
50717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // in the general case,
50817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // so we specifically ask for ordering by size. The
50917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // expectedAttachmentSizes array must
510976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            // be kept sorted by size (ascending) for this test to work properly
51117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            c = mMockContext.getContentResolver().query(Attachment.CONTENT_URI,
512845db9711917ad8936926fc7a1a179fade8e00d5Tony Mantler                    Attachment.CONTENT_PROJECTION, AttachmentColumns.MESSAGE_KEY + "=?",
513845db9711917ad8936926fc7a1a179fade8e00d5Tony Mantler                    new String[] {String.valueOf(message3Id)}, AttachmentColumns.SIZE);
514976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            int numAtts = c.getCount();
515976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            assertEquals(3, numAtts);
516976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            int i = 0;
517976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            while (c.moveToNext()) {
5182f288864b621cfb5aee44eda27df463460d33dd3Tony Mantler                Attachment actual = EmailContent.getContent(mMockContext, c, Attachment.class);
5194119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler                ProviderTestUtils.assertAttachmentEqual("save-message3", atts.get(i), actual);
5204119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler                i++;
5214119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            }
5224119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        } finally {
5234119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            c.close();
5244119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        }
52532bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    }
52632bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
52732bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo
52832bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    @MediumTest
52932bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo    public void testMessageSaveWithJustAttachments() {
53032bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        Account account1 = ProviderTestUtils.setupAccount("message-save", true, mMockContext);
53132bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        long account1Id = account1.mId;
53232bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
53332bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        long box1Id = box1.mId;
53432bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        Cursor c = null;
5354119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler
5364119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        // Message with attachments but no body
53717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message4 = ProviderTestUtils.setupMessage("message4",
53817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
53917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
54017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
54117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
54217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
54332bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        ArrayList<Attachment> atts = new ArrayList<Attachment>();
5444119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        for (int i = 0; i < 3; i++) {
5454119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            atts.add(ProviderTestUtils.setupAttachment(
54617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    -1, expectedAttachmentNames[i], expectedAttachmentSizes[i], false,
54717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    mMockContext));
5484119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        }
5494119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        message4.mAttachments = atts;
5509e2c6bd5f21f2d19eef7ebfe30e6fdf94ede0857Andrew Stadler        message4.save(mMockContext);
5514119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        long message4Id = message4.mId;
5524119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler
55317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Now check the attachments; there should be three and they should
55417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // match name and size
5554119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        c = null;
55641f775bbc731b25854da9f5a80c66a4aa1eb0af9satok
5574119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        try {
55817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // Note that there is NO guarantee of the order of returned records
55917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // in the general case,
56017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // so we specifically ask for ordering by size. The
56117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // expectedAttachmentSizes array must
5624119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            // be kept sorted by size (ascending) for this test to work properly
56317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            c = mMockContext.getContentResolver().query(Attachment.CONTENT_URI,
564845db9711917ad8936926fc7a1a179fade8e00d5Tony Mantler                    Attachment.CONTENT_PROJECTION, AttachmentColumns.MESSAGE_KEY + "=?",
565845db9711917ad8936926fc7a1a179fade8e00d5Tony Mantler                    new String[] {String.valueOf(message4Id)}, AttachmentColumns.SIZE);
5664119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            int numAtts = c.getCount();
5674119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            assertEquals(3, numAtts);
5684119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            int i = 0;
5694119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            while (c.moveToNext()) {
5702f288864b621cfb5aee44eda27df463460d33dd3Tony Mantler                Attachment actual = EmailContent.getContent(mMockContext, c, Attachment.class);
5714119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler                ProviderTestUtils.assertAttachmentEqual("save-message4", atts.get(i), actual);
572976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                i++;
573976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            }
574976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        } finally {
575976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            c.close();
5767143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        }
57744144d0301213536b02f2080237b63d8271b2d0bMihai Preda
57844144d0301213536b02f2080237b63d8271b2d0bMihai Preda        // test EmailContent.restoreAttachmentsWitdMessageId()
57944144d0301213536b02f2080237b63d8271b2d0bMihai Preda        Attachment[] attachments =
58017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                Attachment.restoreAttachmentsWithMessageId(mMockContext, message4Id);
58144144d0301213536b02f2080237b63d8271b2d0bMihai Preda        int size = attachments.length;
58244144d0301213536b02f2080237b63d8271b2d0bMihai Preda        assertEquals(3, size);
58344144d0301213536b02f2080237b63d8271b2d0bMihai Preda        for (int i = 0; i < size; ++i) {
58444144d0301213536b02f2080237b63d8271b2d0bMihai Preda            ProviderTestUtils.assertAttachmentEqual("save-message4", atts.get(i), attachments[i]);
58544144d0301213536b02f2080237b63d8271b2d0bMihai Preda        }
5867143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
587758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
5887143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
589e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank     * Test that saving a message creates the proper snippet for that message
590e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank     */
591e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank    public void testMessageSaveAddsSnippet() {
592e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        Account account = ProviderTestUtils.setupAccount("message-snippet", true, mMockContext);
593e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        Mailbox box = ProviderTestUtils.setupMailbox("box1", account.mId, true, mMockContext);
594e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank
595e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        // Create a message without a body, unsaved
59617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message = ProviderTestUtils.setupMessage("message",
59717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account.mId,
59817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box.mId,
59917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
60017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
60117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
602e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        message.mText = "This is some text";
603e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        message.mHtml = "<html>This is some text</html>";
604e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        message.save(mMockContext);
605e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        Message restoredMessage = Message.restoreMessageWithId(mMockContext, message.mId);
606e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        // We should have the plain text as the snippet
60717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(
60817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                restoredMessage.mSnippet, TextUtilities.makeSnippetFromPlainText(message.mText));
609e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank
610e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        // Start again
61117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        message = ProviderTestUtils.setupMessage("message",
61217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account.mId,
61317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box.mId,
61417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
61517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
61617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
617e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        message.mText = null;
618e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        message.mHtml = "<html>This is some text</html>";
619e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        message.save(mMockContext);
620e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        restoredMessage = Message.restoreMessageWithId(mMockContext, message.mId);
621e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank        // We should have the plain text as the snippet
62217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(
62317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                restoredMessage.mSnippet, TextUtilities.makeSnippetFromHtmlText(message.mHtml));
624e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank    }
625e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank
626e7b9e4ab94093127002f4cddd3ffcc3deb1c0b41Marc Blank    /**
6277143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: update account
6287143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
629758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
6307143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
6317143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: update mailbox
6327143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
633758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
6347143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
6357143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: update message
6367143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
637758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
6387143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
6397143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * Test delete account
6407143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: hostauth
6417143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
6427143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public void testAccountDelete() {
64328448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Account account1 = ProviderTestUtils.setupAccount("account-delete-1", true, mMockContext);
6447143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long account1Id = account1.mId;
64528448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Account account2 = ProviderTestUtils.setupAccount("account-delete-2", true, mMockContext);
6467143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long account2Id = account2.mId;
6477143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
6487143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // make sure there are two accounts
6497143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        int numBoxes = EmailContent.count(mMockContext, Account.CONTENT_URI, null, null);
6507143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(2, numBoxes);
6517143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
6527143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // now delete one of them
6537143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        Uri uri = ContentUris.withAppendedId(Account.CONTENT_URI, account1Id);
6547143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
6557143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
6567143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // make sure there's only one account now
6577143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        numBoxes = EmailContent.count(mMockContext, Account.CONTENT_URI, null, null);
6587143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(1, numBoxes);
6597143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
6607143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // now delete the other one
6617143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        uri = ContentUris.withAppendedId(Account.CONTENT_URI, account2Id);
6627143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
6637143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
6647143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // make sure there are no accounts now
6657143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        numBoxes = EmailContent.count(mMockContext, Account.CONTENT_URI, null, null);
6667143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(0, numBoxes);
6677143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
668758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
6697143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
6709627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * Test for Body.lookupBodyIdWithMessageId()
6719627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * Verifies that:
6729627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * - for a message without body, -1 is returned.
6739627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * - for a mesage with body, the id matches the one from loadBodyForMessageId.
6749627d014e16235eadf981b9165807dc72a14a383Mihai Preda     */
6759627d014e16235eadf981b9165807dc72a14a383Mihai Preda    public void testLookupBodyIdWithMessageId() {
6769627d014e16235eadf981b9165807dc72a14a383Mihai Preda        final ContentResolver resolver = mMockContext.getContentResolver();
6779627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Account account1 = ProviderTestUtils.setupAccount("orphaned body", true, mMockContext);
6789627d014e16235eadf981b9165807dc72a14a383Mihai Preda        long account1Id = account1.mId;
6799627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
6809627d014e16235eadf981b9165807dc72a14a383Mihai Preda        long box1Id = box1.mId;
6819627d014e16235eadf981b9165807dc72a14a383Mihai Preda
6829627d014e16235eadf981b9165807dc72a14a383Mihai Preda        // 1. create message with no body, check that returned bodyId is -1
68317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
68417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
68517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
68617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
68717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
68817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
6899627d014e16235eadf981b9165807dc72a14a383Mihai Preda        long message1Id = message1.mId;
690bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        long bodyId1 = Body.lookupBodyIdWithMessageId(mMockContext, message1Id);
6919627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertEquals(bodyId1, -1);
6929627d014e16235eadf981b9165807dc72a14a383Mihai Preda
6939627d014e16235eadf981b9165807dc72a14a383Mihai Preda        // 2. create message with body, check that returned bodyId is correct
69417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message1",
69517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
69617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
69717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
69817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
69917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
7009627d014e16235eadf981b9165807dc72a14a383Mihai Preda        long message2Id = message2.mId;
701bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        long bodyId2 = Body.lookupBodyIdWithMessageId(mMockContext, message2Id);
7029627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Body body = loadBodyForMessageId(message2Id);
7039627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertNotNull(body);
7049627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertEquals(body.mId, bodyId2);
7059627d014e16235eadf981b9165807dc72a14a383Mihai Preda    }
7069627d014e16235eadf981b9165807dc72a14a383Mihai Preda
7079627d014e16235eadf981b9165807dc72a14a383Mihai Preda    /**
7089627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * Test for Body.updateBodyWithMessageId().
7099627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * 1. - create message without body,
7109627d014e16235eadf981b9165807dc72a14a383Mihai Preda     *    - update its body (set TEXT_CONTENT)
7119627d014e16235eadf981b9165807dc72a14a383Mihai Preda     *    - check correct updated body is read back
7129627d014e16235eadf981b9165807dc72a14a383Mihai Preda     *
7139627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * 2. - create message with body,
7149627d014e16235eadf981b9165807dc72a14a383Mihai Preda     *    - update body (set TEXT_CONTENT)
7159627d014e16235eadf981b9165807dc72a14a383Mihai Preda     *    - check correct updated body is read back
7169627d014e16235eadf981b9165807dc72a14a383Mihai Preda     */
7179627d014e16235eadf981b9165807dc72a14a383Mihai Preda    public void testUpdateBodyWithMessageId() {
7189627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Account account1 = ProviderTestUtils.setupAccount("orphaned body", true, mMockContext);
7199627d014e16235eadf981b9165807dc72a14a383Mihai Preda        long account1Id = account1.mId;
7209627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
7219627d014e16235eadf981b9165807dc72a14a383Mihai Preda        long box1Id = box1.mId;
7229627d014e16235eadf981b9165807dc72a14a383Mihai Preda
7239627d014e16235eadf981b9165807dc72a14a383Mihai Preda        final String textContent = "foobar some odd text";
724936babc145e2e6eb2e222f2ce5e3da8f9e4fb9f2Andrew Stadler        final String htmlContent = "and some html";
7259627d014e16235eadf981b9165807dc72a14a383Mihai Preda
7269627d014e16235eadf981b9165807dc72a14a383Mihai Preda        ContentValues values = new ContentValues();
7279627d014e16235eadf981b9165807dc72a14a383Mihai Preda        values.put(BodyColumns.TEXT_CONTENT, textContent);
728936babc145e2e6eb2e222f2ce5e3da8f9e4fb9f2Andrew Stadler        values.put(BodyColumns.HTML_CONTENT, htmlContent);
729e25698319325d5e9619bc45ae85d861b0f1c5954Marc Blank        values.put(BodyColumns.SOURCE_MESSAGE_KEY, 17);
7309627d014e16235eadf981b9165807dc72a14a383Mihai Preda
7319627d014e16235eadf981b9165807dc72a14a383Mihai Preda        // 1
73217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
73317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
73417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
73517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
73617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
73717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
7389627d014e16235eadf981b9165807dc72a14a383Mihai Preda        long message1Id = message1.mId;
7399627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Body body1 = loadBodyForMessageId(message1Id);
7409627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertNull(body1);
7419627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Body.updateBodyWithMessageId(mMockContext, message1Id, values);
7429627d014e16235eadf981b9165807dc72a14a383Mihai Preda        body1 = loadBodyForMessageId(message1Id);
7439627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertNotNull(body1);
7449627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertEquals(body1.mTextContent, textContent);
745936babc145e2e6eb2e222f2ce5e3da8f9e4fb9f2Andrew Stadler        assertEquals(body1.mHtmlContent, htmlContent);
746e25698319325d5e9619bc45ae85d861b0f1c5954Marc Blank        assertEquals(body1.mSourceKey, 17);
7479627d014e16235eadf981b9165807dc72a14a383Mihai Preda
7489627d014e16235eadf981b9165807dc72a14a383Mihai Preda        // 2
74917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message1",
75017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
75117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
75217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
75317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
75417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
7559627d014e16235eadf981b9165807dc72a14a383Mihai Preda        long message2Id = message2.mId;
7569627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Body body2 = loadBodyForMessageId(message2Id);
7579627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertNotNull(body2);
7589627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertTrue(!body2.mTextContent.equals(textContent));
7599627d014e16235eadf981b9165807dc72a14a383Mihai Preda        Body.updateBodyWithMessageId(mMockContext, message2Id, values);
7609627d014e16235eadf981b9165807dc72a14a383Mihai Preda        body2 = loadBodyForMessageId(message1Id);
7619627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertNotNull(body2);
7629627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertEquals(body2.mTextContent, textContent);
763936babc145e2e6eb2e222f2ce5e3da8f9e4fb9f2Andrew Stadler        assertEquals(body2.mHtmlContent, htmlContent);
764e25698319325d5e9619bc45ae85d861b0f1c5954Marc Blank        assertEquals(body2.mSourceKey, 17);
7659627d014e16235eadf981b9165807dc72a14a383Mihai Preda    }
7669627d014e16235eadf981b9165807dc72a14a383Mihai Preda
7679627d014e16235eadf981b9165807dc72a14a383Mihai Preda    /**
768c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler     * Test body retrieve methods
769c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler     */
770c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler    public void testBodyRetrieve() {
771c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler        // No account needed
772c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler        // No mailbox needed
77317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 =
77417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupMessage("bodyretrieve", 1, 1, true, true, mMockContext);
775c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler        long messageId = message1.mId;
776c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler
77717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(message1.mText, Body.restoreBodyTextWithMessageId(mMockContext, messageId));
77817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(message1.mHtml, Body.restoreBodyHtmlWithMessageId(mMockContext, messageId));
77917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(message1.mSourceKey, Body.restoreBodySourceKey(mMockContext, messageId));
780c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler    }
781c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler
782c41c47fa07a22f8a7612fb0191f152a36d95b7a5Andrew Stadler    /**
783fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * Test delete body.
784fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 1. create message without body (message id 1)
785fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 2. create message with body (message id 2. The body has _id 1 and messageKey 2).
786fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 3. delete first message.
787fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 4. verify that body for message 2 has not been deleted.
788fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 5. delete message 2, verify body is deleted.
789fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     */
790fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    public void testDeleteBody() {
791fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        final ContentResolver resolver = mMockContext.getContentResolver();
792fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
793fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // Create account and mailboxes
794fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        Account account1 = ProviderTestUtils.setupAccount("orphaned body", true, mMockContext);
795fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long account1Id = account1.mId;
796fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
797fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long box1Id = box1.mId;
798fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
799fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 1. create message without body
80017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
80117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
80217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
80317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
80417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
80517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
806fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long message1Id = message1.mId;
807fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
808fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 2. create message with body
80917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message1",
81017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
81117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
81217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
81317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
81417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
815fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long message2Id = message2.mId;
816fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // verify body is there
817fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        assertNotNull(loadBodyForMessageId(message2Id));
818fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
819fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 3. delete first message
820fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        resolver.delete(ContentUris.withAppendedId(Message.CONTENT_URI, message1Id), null, null);
821c0c9c33322deecace00a32766e0a1b355aad4b31Marc Blank
822fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 4. verify body for second message wasn't deleted
823fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        assertNotNull(loadBodyForMessageId(message2Id));
824fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
825fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 5. delete second message, check its body is deleted
826fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        resolver.delete(ContentUris.withAppendedId(Message.CONTENT_URI, message2Id), null, null);
827fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        assertNull(loadBodyForMessageId(message2Id));
828fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    }
829fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
830fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    /**
831fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * Test delete orphan bodies.
832fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 1. create message without body (message id 1)
833fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 2. create message with body (message id 2. Body has _id 1 and messageKey 2).
834fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 3. delete first message.
835fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 4. delete some other mailbox -- this triggers delete orphan bodies.
836fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     * 5. verify that body for message 2 has not been deleted.
837fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda     */
838fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    public void testDeleteOrphanBodies() {
839fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        final ContentResolver resolver = mMockContext.getContentResolver();
840fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
841ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Create account and two mailboxes
842fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        Account account1 = ProviderTestUtils.setupAccount("orphaned body", true, mMockContext);
843fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long account1Id = account1.mId;
844fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
845fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long box1Id = box1.mId;
846fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        Mailbox box2 = ProviderTestUtils.setupMailbox("box2", account1Id, true, mMockContext);
847fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long box2Id = box2.mId;
848fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
849fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 1. create message without body
85017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
85117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
85217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
85317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
85417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
85517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
856fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long message1Id = message1.mId;
857fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
858fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 2. create message with body
85917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message1",
86017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
86117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
86217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
86317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
86417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
865fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        long message2Id = message2.mId;
86617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // verify body is there
867fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        assertNotNull(loadBodyForMessageId(message2Id));
868fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
869fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 3. delete first message
870fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        resolver.delete(ContentUris.withAppendedId(Message.CONTENT_URI, message1Id), null, null);
871fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
872fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        // 4. delete some mailbox (because it triggers "delete orphan bodies")
873fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        resolver.delete(ContentUris.withAppendedId(Mailbox.CONTENT_URI, box2Id), null, null);
874fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
87517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // 5. verify body for second message wasn't deleted during
87617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // "delete orphan bodies"
877fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda        assertNotNull(loadBodyForMessageId(message2Id));
878fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    }
879fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda
880fb7974f5bfb6275fb856b0f7ff386ef10680c83aMihai Preda    /**
8810efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank     * Note that we can't use EmailContent.count() here because it uses a projection including
8820efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank     * count(*), and count(*) is incompatible with a LIMIT (i.e. the limit would be applied to the
8830efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank     * single column returned with count(*), rather than to the query itself)
8840efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank     */
8850efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank    private int count(Context context, Uri uri, String selection, String[] selectionArgs) {
88617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Cursor c = context.getContentResolver()
88717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                .query(uri, EmailContent.ID_PROJECTION, selection, selectionArgs, null);
8880efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        try {
8890efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank            return c.getCount();
8900efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        } finally {
8910efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank            c.close();
8920efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        }
8930efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank    }
8940efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank
8950efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank    public void testMessageQueryWithLimit() {
8960efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        final Context context = mMockContext;
8970efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank
8980efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        // Create account and two mailboxes
8990efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        Account acct = ProviderTestUtils.setupAccount("orphaned body", true, context);
9000efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", acct.mId, true, context);
9010efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        Mailbox box2 = ProviderTestUtils.setupMailbox("box2", acct.mId, true, context);
9020efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank
9030efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        // Create 4 messages in box1
9040efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        ProviderTestUtils.setupMessage("message1", acct.mId, box1.mId, false, true, context);
9050efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        ProviderTestUtils.setupMessage("message2", acct.mId, box1.mId, false, true, context);
9060efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        ProviderTestUtils.setupMessage("message3", acct.mId, box1.mId, false, true, context);
9070efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        ProviderTestUtils.setupMessage("message4", acct.mId, box1.mId, false, true, context);
9080efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank
9090efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        // Create 4 messages in box2
9100efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        ProviderTestUtils.setupMessage("message1", acct.mId, box2.mId, false, true, context);
9110efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        ProviderTestUtils.setupMessage("message2", acct.mId, box2.mId, false, true, context);
9120efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        ProviderTestUtils.setupMessage("message3", acct.mId, box2.mId, false, true, context);
9130efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        ProviderTestUtils.setupMessage("message4", acct.mId, box2.mId, false, true, context);
9140efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank
9150efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        // Check normal case, special case (limit 1), and arbitrary limits
9160efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        assertEquals(8, count(mMockContext, Message.CONTENT_URI, null, null));
91717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(1,
91817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                count(mMockContext, EmailContent.uriWithLimit(Message.CONTENT_URI, 1), null, null));
91917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(3,
92017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                count(mMockContext, EmailContent.uriWithLimit(Message.CONTENT_URI, 3), null, null));
9210efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        assertEquals(8, count(mMockContext, EmailContent.uriWithLimit(Message.CONTENT_URI, 100),
9220efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank                null, null));
9230efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank
9240efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        // Check that it works with selection/selection args
9250efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank        String[] args = new String[] {Long.toString(box1.mId)};
92617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(4,
92717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                count(mMockContext, Message.CONTENT_URI, MessageColumns.MAILBOX_KEY + "=?", args));
92817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(1, count(mMockContext, EmailContent.uriWithLimit(Message.CONTENT_URI, 1),
9290efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank                MessageColumns.MAILBOX_KEY + "=?", args));
9300efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank    }
9310efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank
9320efe738e05a31e0c1ebfba645bd2364a373a3f33Marc Blank    /**
933ef83299b99288c00b9d661260d19715e73e6889cMarc Blank     * Test delete orphan messages
934ef83299b99288c00b9d661260d19715e73e6889cMarc Blank     * 1. create message without body (message id 1)
935ef83299b99288c00b9d661260d19715e73e6889cMarc Blank     * 2. create message with body (message id 2. Body has _id 1 and messageKey 2).
936ef83299b99288c00b9d661260d19715e73e6889cMarc Blank     * 3. delete first message.
937ef83299b99288c00b9d661260d19715e73e6889cMarc Blank     * 4. delete some other mailbox -- this triggers delete orphan bodies.
938ef83299b99288c00b9d661260d19715e73e6889cMarc Blank     * 5. verify that body for message 2 has not been deleted.
939ef83299b99288c00b9d661260d19715e73e6889cMarc Blank     */
94017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    public void testDeleteOrphanMessages() {
941ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        final ContentResolver resolver = mMockContext.getContentResolver();
942ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        final Context context = mMockContext;
943ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
944ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Create account and two mailboxes
945ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        Account acct = ProviderTestUtils.setupAccount("orphaned body", true, context);
946ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", acct.mId, true, context);
947ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        Mailbox box2 = ProviderTestUtils.setupMailbox("box2", acct.mId, true, context);
948ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
949ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Create 4 messages in box1
95017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msg1_1 = ProviderTestUtils.setupMessage("message1",
95117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
95217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1.mId,
95317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
95417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
95517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
95617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msg1_2 = ProviderTestUtils.setupMessage("message2",
95717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
95817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1.mId,
95917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
96017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
96117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
96217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msg1_3 = ProviderTestUtils.setupMessage("message3",
96317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
96417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1.mId,
96517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
96617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
96717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
96817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msg1_4 = ProviderTestUtils.setupMessage("message4",
96917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
97017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1.mId,
97117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
97217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
97317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
974ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
975ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Create 4 messages in box2
97617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msg2_1 = ProviderTestUtils.setupMessage("message1",
97717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
97817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box2.mId,
97917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
98017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
98117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
98217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msg2_2 = ProviderTestUtils.setupMessage("message2",
98317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
98417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box2.mId,
98517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
98617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
98717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
98817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msg2_3 = ProviderTestUtils.setupMessage("message3",
98917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
99017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box2.mId,
99117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
99217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
99317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
99417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msg2_4 = ProviderTestUtils.setupMessage("message4",
99517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
99617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box2.mId,
99717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
99817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
99917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
1000ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1001ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Delete 2 from each mailbox
100217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.delete(
100317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, msg1_1.mId), null, null);
100417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.delete(
100517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, msg1_2.mId), null, null);
100617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.delete(
100717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, msg2_1.mId), null, null);
100817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.delete(
100917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, msg2_2.mId), null, null);
1010ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1011ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // There should be 4 items in the deleted item table
1012ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(4, EmailContent.count(context, Message.DELETED_CONTENT_URI, null, null));
1013ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1014ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Update 2 from each mailbox
1015ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        ContentValues v = new ContentValues();
1016ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        v.put(MessageColumns.DISPLAY_NAME, "--updated--");
101717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.update(
101817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, msg1_3.mId), v, null, null);
101917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.update(
102017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, msg1_4.mId), v, null, null);
102117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.update(
102217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, msg2_3.mId), v, null, null);
102317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.update(
102417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, msg2_4.mId), v, null, null);
102517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie
102617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // There should be 4 items in the updated item table
1027ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(4, EmailContent.count(context, Message.UPDATED_CONTENT_URI, null, null));
1028ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
102917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Manually add 2 messages from a "deleted" mailbox to deleted and
103017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // updated tables
1031ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Use a value > 2 for the deleted box id
1032ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        long delBoxId = 10;
1033ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Create 4 messages in the "deleted" mailbox
103417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msgX_A = ProviderTestUtils.setupMessage("messageA",
103517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
103617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                delBoxId,
103717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
103817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
103917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
104017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msgX_B = ProviderTestUtils.setupMessage("messageB",
104117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
104217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                delBoxId,
104317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
104417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
104517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
104617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msgX_C = ProviderTestUtils.setupMessage("messageC",
104717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
104817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                delBoxId,
104917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
105017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
105117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
105217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message msgX_D = ProviderTestUtils.setupMessage("messageD",
105317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                acct.mId,
105417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                delBoxId,
105517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
105617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
105717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                context);
1058ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1059ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        ContentValues cv;
106017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // We have to assign id's manually because there are no autoincrement
106117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // id's for these tables
106217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Start with an id that won't exist, since id's in these tables must be
106317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // unique
1064ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        long msgId = 10;
106517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // It's illegal to manually insert these, so we need to catch the
106617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // exception
1067ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // NOTE: The insert succeeds, and then throws the exception
1068ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        try {
1069ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            cv = msgX_A.toContentValues();
1070ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            cv.put(EmailContent.RECORD_ID, msgId++);
1071ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            resolver.insert(Message.DELETED_CONTENT_URI, cv);
1072ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        } catch (IllegalArgumentException e) {
1073ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        }
1074ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        try {
1075ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            cv = msgX_B.toContentValues();
1076ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            cv.put(EmailContent.RECORD_ID, msgId++);
1077ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            resolver.insert(Message.DELETED_CONTENT_URI, cv);
1078ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        } catch (IllegalArgumentException e) {
1079ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        }
1080ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        try {
1081ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            cv = msgX_C.toContentValues();
1082ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            cv.put(EmailContent.RECORD_ID, msgId++);
1083ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            resolver.insert(Message.UPDATED_CONTENT_URI, cv);
1084ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        } catch (IllegalArgumentException e) {
1085ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        }
1086ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        try {
1087ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            cv = msgX_D.toContentValues();
1088ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            cv.put(EmailContent.RECORD_ID, msgId++);
1089ef83299b99288c00b9d661260d19715e73e6889cMarc Blank            resolver.insert(Message.UPDATED_CONTENT_URI, cv);
1090ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        } catch (IllegalArgumentException e) {
1091ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        }
1092ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1093ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // There should be 6 items in the deleted and updated tables
1094ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(6, EmailContent.count(context, Message.UPDATED_CONTENT_URI, null, null));
1095ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(6, EmailContent.count(context, Message.DELETED_CONTENT_URI, null, null));
1096ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1097ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Delete the orphans
109817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        EmailProvider.deleteMessageOrphans(
109917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                getProvider().getDatabase(context), Message.DELETED_TABLE_NAME);
110017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        EmailProvider.deleteMessageOrphans(
110117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                getProvider().getDatabase(context), Message.UPDATED_TABLE_NAME);
1102ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
110317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // There should now be 4 messages in each of the deleted and updated
110417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // tables again
1105ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(4, EmailContent.count(context, Message.UPDATED_CONTENT_URI, null, null));
1106ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(4, EmailContent.count(context, Message.DELETED_CONTENT_URI, null, null));
1107ef83299b99288c00b9d661260d19715e73e6889cMarc Blank    }
1108ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1109ef83299b99288c00b9d661260d19715e73e6889cMarc Blank    /**
11107143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * Test delete message
11117143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: body
11127143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: attachments
11137143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
11147143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public void testMessageDelete() {
111528448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Account account1 = ProviderTestUtils.setupAccount("message-delete", true, mMockContext);
11167143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long account1Id = account1.mId;
111728448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
11187143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long box1Id = box1.mId;
111917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
112017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
112117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
112217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
112317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
112417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
11257143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long message1Id = message1.mId;
112617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message2",
112717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
112817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
112917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
113017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
113117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
11327143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long message2Id = message2.mId;
11337143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
113417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String selection = EmailContent.MessageColumns.ACCOUNT_KEY + "=? AND "
113517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                + EmailContent.MessageColumns.MAILBOX_KEY + "=?";
113617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String[] selArgs = new String[] {String.valueOf(account1Id), String.valueOf(box1Id)};
11377143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
11387143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // make sure there are two messages
11397143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        int numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
11407143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(2, numMessages);
11417143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
11427143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // now delete one of them
11437143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        Uri uri = ContentUris.withAppendedId(Message.CONTENT_URI, message1Id);
11447143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
11457143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
11467143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // make sure there's only one message now
11477143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
11487143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(1, numMessages);
11497143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
11507143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // now delete the other one
11517143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        uri = ContentUris.withAppendedId(Message.CONTENT_URI, message2Id);
11527143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
11537143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
11547143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // make sure there are no messages now
11557143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
11567143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(0, numMessages);
11577143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
1158758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
11597143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
1160ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler     * Test delete synced message
1161ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler     * TODO: body
1162ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler     * TODO: attachments
1163ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler     */
1164ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler    public void testSyncedMessageDelete() {
116517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Account account1 =
116617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupAccount("synced-message-delete", true, mMockContext);
1167ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        long account1Id = account1.mId;
116828448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
1169ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        long box1Id = box1.mId;
117017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
117117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
117217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
117317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
117417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
117517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
1176ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        long message1Id = message1.mId;
117717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message2",
117817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
117917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
118017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
118117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
118217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
1183ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        long message2Id = message2.mId;
1184ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1185ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        String selection = EmailContent.MessageColumns.ACCOUNT_KEY + "=? AND "
1186ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler                + EmailContent.MessageColumns.MAILBOX_KEY + "=?";
118717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String[] selArgs = new String[] {String.valueOf(account1Id), String.valueOf(box1Id)};
1188ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1189ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure there are two messages
1190ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        int numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
1191ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(2, numMessages);
1192ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1193ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure we start with no synced deletions
119417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
119517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.DELETED_CONTENT_URI, selection, selArgs);
1196ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(0, numMessages);
1197ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1198ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // now delete one of them SYNCED
1199ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        Uri uri = ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, message1Id);
1200ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
1201ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1202ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure there's only one message now
1203ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
1204ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(1, numMessages);
1205ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1206ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure there's one synced deletion now
120717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
120817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.DELETED_CONTENT_URI, selection, selArgs);
1209ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(1, numMessages);
1210ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1211ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // now delete the other one NOT SYNCED
1212ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        uri = ContentUris.withAppendedId(Message.CONTENT_URI, message2Id);
1213ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
1214ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1215ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure there are no messages now
1216ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
1217ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(0, numMessages);
1218ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1219ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure there's still one deletion now
122017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
122117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.DELETED_CONTENT_URI, selection, selArgs);
1222ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(1, numMessages);
1223ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler    }
1224ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1225ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler    /**
1226ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler     * Test message update
1227ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler     * TODO: body
1228ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler     * TODO: attachments
1229ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler     */
1230ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler    public void testMessageUpdate() {
123128448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Account account1 = ProviderTestUtils.setupAccount("message-update", true, mMockContext);
1232ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        long account1Id = account1.mId;
123328448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
1234ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        long box1Id = box1.mId;
123517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
123617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
123717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
123817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
123917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
124017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
1241ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        long message1Id = message1.mId;
124217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message2",
124317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
124417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
124517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
124617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
124717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
1248ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        long message2Id = message2.mId;
1249ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        ContentResolver cr = mMockContext.getContentResolver();
1250ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1251ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        String selection = EmailContent.MessageColumns.ACCOUNT_KEY + "=? AND "
1252ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler                + EmailContent.MessageColumns.MAILBOX_KEY + "=?";
125317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String[] selArgs = new String[] {String.valueOf(account1Id), String.valueOf(box1Id)};
1254ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1255ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure there are two messages
1256ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        int numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
1257ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(2, numMessages);
1258ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1259ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // change the first one
1260ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        Uri uri = ContentUris.withAppendedId(Message.CONTENT_URI, message1Id);
1261ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        ContentValues cv = new ContentValues();
1262ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        cv.put(MessageColumns.FROM_LIST, "from-list");
1263ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        cr.update(uri, cv, null, null);
1264ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1265ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure there's no updated message
126617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
126717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.UPDATED_CONTENT_URI, selection, selArgs);
1268ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(0, numMessages);
1269ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1270ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // get the message back from the provider, make sure the change "stuck"
1271ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        Message restoredMessage = Message.restoreMessageWithId(mMockContext, message1Id);
1272ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals("from-list", restoredMessage.mFrom);
1273ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1274ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // change the second one
1275ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        uri = ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, message2Id);
1276ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        cv = new ContentValues();
1277ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        cv.put(MessageColumns.FROM_LIST, "from-list");
1278ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        cr.update(uri, cv, null, null);
1279ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1280ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // make sure there's one updated message
128117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
128217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.UPDATED_CONTENT_URI, selection, selArgs);
1283ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(1, numMessages);
1284ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1285ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // get the message back from the provider, make sure the change "stuck",
1286ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // as before
1287ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        restoredMessage = Message.restoreMessageWithId(mMockContext, message2Id);
1288ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals("from-list", restoredMessage.mFrom);
1289ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1290ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // get the original message back from the provider
129117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Cursor c =
129217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                cr.query(Message.UPDATED_CONTENT_URI, Message.CONTENT_PROJECTION, null, null, null);
1293ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        try {
1294ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler            assertTrue(c.moveToFirst());
12952f288864b621cfb5aee44eda27df463460d33dd3Tony Mantler            Message originalMessage = EmailContent.getContent(mMockContext, c, Message.class);
1296ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler            // make sure this has the original value
1297ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler            assertEquals("from message2", originalMessage.mFrom);
1298ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler            // Should only be one
1299ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler            assertFalse(c.moveToNext());
1300ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        } finally {
1301ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler            c.close();
1302ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        }
1303ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1304ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // delete the second message
1305ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        cr.delete(ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, message2Id), null, null);
1306ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1307ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // hey, presto! the change should be gone
130817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
130917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.UPDATED_CONTENT_URI, selection, selArgs);
1310ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(0, numMessages);
1311ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1312ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        // and there should now be a deleted record
131317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
131417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.DELETED_CONTENT_URI, selection, selArgs);
1315ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler        assertEquals(1, numMessages);
1316ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler    }
1317ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler
1318ddc871d9eddd088c0200a30dbfeb24812a81cf80Andrew Stadler    /**
13197143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: cascaded delete account
13207143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: hostauth
13217143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: body
13227143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: attachments
13237143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: create other account, mailbox & messages and confirm the right objects were deleted
13247143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
13257143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public void testCascadeDeleteAccount() {
132617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Account account1 =
132717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupAccount("account-delete-cascade", true, mMockContext);
13287143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long account1Id = account1.mId;
132928448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
13307143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long box1Id = box1.mId;
133117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        /* Message message1 = */ProviderTestUtils.setupMessage("message1",
133217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
133317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
133417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
133517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
133617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
133717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        /* Message message2 = */ProviderTestUtils.setupMessage("message2",
133817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
133917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
134017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
134117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
134217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
13437143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
13447143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // make sure there is one account, one mailbox, and two messages
13457143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        int numAccounts = EmailContent.count(mMockContext, Account.CONTENT_URI, null, null);
13467143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(1, numAccounts);
13477143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        int numBoxes = EmailContent.count(mMockContext, Mailbox.CONTENT_URI, null, null);
13487143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(1, numBoxes);
13497143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        int numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, null, null);
13507143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(2, numMessages);
13517143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
13527143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // delete the account
13537143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        Uri uri = ContentUris.withAppendedId(Account.CONTENT_URI, account1Id);
13547143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
13557143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
13567143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // make sure there are no accounts, mailboxes, or messages
13577143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        numAccounts = EmailContent.count(mMockContext, Account.CONTENT_URI, null, null);
13587143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(0, numAccounts);
13597143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        numBoxes = EmailContent.count(mMockContext, Mailbox.CONTENT_URI, null, null);
13607143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(0, numBoxes);
13617143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, null, null);
13627143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(0, numMessages);
13637143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
1364758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
13657143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
13667143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * Test cascaded delete mailbox
13677143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: body
13687143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: attachments
13697143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     * TODO: create other mailbox & messages and confirm the right objects were deleted
13707143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
13717143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    public void testCascadeDeleteMailbox() {
137217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Account account1 =
137317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupAccount("mailbox-delete-cascade", true, mMockContext);
13747143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long account1Id = account1.mId;
137528448e782b825d1978c0923003a2cf91efe733dcAndrew Stadler        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
13767143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        long box1Id = box1.mId;
137717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
137817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
137917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
138017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
138117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
138217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
138317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message2",
138417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
138517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
138617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
138717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
138817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
138917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message3 = ProviderTestUtils.setupMessage("message3",
139017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
139117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
139217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
139317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
139417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
139517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message4 = ProviderTestUtils.setupMessage("message4",
139617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
139717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
139817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
139917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
140017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
1401ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        ProviderTestUtils.setupMessage("message5", account1Id, box1Id, false, true, mMockContext);
1402ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        ProviderTestUtils.setupMessage("message6", account1Id, box1Id, false, true, mMockContext);
14037143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
140417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String selection = EmailContent.MessageColumns.ACCOUNT_KEY + "=? AND "
140517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                + EmailContent.MessageColumns.MAILBOX_KEY + "=?";
140617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String[] selArgs = new String[] {String.valueOf(account1Id), String.valueOf(box1Id)};
14077143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
14080e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // make sure there are six messages
14097143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        int numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
1410ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(6, numMessages);
1411ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1412ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        ContentValues cv = new ContentValues();
1413845db9711917ad8936926fc7a1a179fade8e00d5Tony Mantler        cv.put(MessageColumns.SERVER_ID, "SERVER_ID");
1414ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        ContentResolver resolver = mMockContext.getContentResolver();
1415ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
1416ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // Update two messages
141717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.update(ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, message1.mId), cv,
1418ef83299b99288c00b9d661260d19715e73e6889cMarc Blank                null, null);
141917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.update(ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, message2.mId), cv,
1420ef83299b99288c00b9d661260d19715e73e6889cMarc Blank                null, null);
142117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Delete two messages
142217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.delete(
142317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, message3.mId), null, null);
142417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        resolver.delete(
142517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ContentUris.withAppendedId(Message.SYNCED_CONTENT_URI, message4.mId), null, null);
1426ef83299b99288c00b9d661260d19715e73e6889cMarc Blank
142717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // There should now be two messages in updated/deleted, and 4 in
142817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // messages
1429ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
1430ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(4, numMessages);
143117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
143217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.DELETED_CONTENT_URI, selection, selArgs);
1433ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(2, numMessages);
143417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
143517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.UPDATED_CONTENT_URI, selection, selArgs);
14367143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(2, numMessages);
1437758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
14387143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        // now delete the mailbox
14397143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        Uri uri = ContentUris.withAppendedId(Mailbox.CONTENT_URI, box1Id);
1440ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        resolver.delete(uri, null, null);
1441758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
1442ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        // there should now be zero messages in all three tables
14437143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        numMessages = EmailContent.count(mMockContext, Message.CONTENT_URI, selection, selArgs);
14447143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler        assertEquals(0, numMessages);
144517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
144617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.DELETED_CONTENT_URI, selection, selArgs);
1447ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(0, numMessages);
144817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numMessages =
144917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                EmailContent.count(mMockContext, Message.UPDATED_CONTENT_URI, selection, selArgs);
1450ef83299b99288c00b9d661260d19715e73e6889cMarc Blank        assertEquals(0, numMessages);
14517143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    }
1452758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
14537143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler    /**
14546c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler     * Test cascaded delete message
14556c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler     * Confirms that deleting a message will also delete its body & attachments
14567143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler     */
14576c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler    public void testCascadeMessageDelete() {
14586c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        Account account1 = ProviderTestUtils.setupAccount("message-cascade", true, mMockContext);
14596c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        long account1Id = account1.mId;
14606c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", account1Id, true, mMockContext);
14616c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        long box1Id = box1.mId;
1462bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki
14636c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        // Each message has a body, and also give each 2 attachments
146417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 = ProviderTestUtils.setupMessage("message1",
146517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
146617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
146717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
146817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
146917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
14706c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        ArrayList<Attachment> atts = new ArrayList<Attachment>();
14716c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        for (int i = 0; i < 2; i++) {
14726c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler            atts.add(ProviderTestUtils.setupAttachment(
147317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    -1, expectedAttachmentNames[i], expectedAttachmentSizes[i], false,
147417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    mMockContext));
14756c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        }
14766c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        message1.mAttachments = atts;
14776c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        message1.save(mMockContext);
14786c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        long message1Id = message1.mId;
14796c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
148017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 = ProviderTestUtils.setupMessage("message2",
148117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                account1Id,
148217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                box1Id,
148317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
148417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                false,
148517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext);
14866c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        atts = new ArrayList<Attachment>();
14876c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        for (int i = 0; i < 2; i++) {
14886c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler            atts.add(ProviderTestUtils.setupAttachment(
148917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    -1, expectedAttachmentNames[i], expectedAttachmentSizes[i], false,
149017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                    mMockContext));
14916c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        }
14926c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        message2.mAttachments = atts;
14936c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        message2.save(mMockContext);
14946c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        long message2Id = message2.mId;
14956c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
149617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Set up to test total counts of bodies & attachments for our test
149717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // messages
14986c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        String bodySelection = BodyColumns.MESSAGE_KEY + " IN (?,?)";
14996c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        String attachmentSelection = AttachmentColumns.MESSAGE_KEY + " IN (?,?)";
150017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String[] selArgs = new String[] {String.valueOf(message1Id), String.valueOf(message2Id)};
1501bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki
15026c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        // make sure there are two bodies
15036c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        int numBodies = EmailContent.count(mMockContext, Body.CONTENT_URI, bodySelection, selArgs);
15046c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        assertEquals(2, numBodies);
15056c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
15066c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        // make sure there are four attachments
150717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        int numAttachments = EmailContent.count(
150817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext, Attachment.CONTENT_URI, attachmentSelection, selArgs);
15096c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        assertEquals(4, numAttachments);
15106c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
15116c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        // now delete one of the messages
15126c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        Uri uri = ContentUris.withAppendedId(Message.CONTENT_URI, message1Id);
15136c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
15146c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
15156c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        // there should be one body and two attachments
15166c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        numBodies = EmailContent.count(mMockContext, Body.CONTENT_URI, bodySelection, selArgs);
15176c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        assertEquals(1, numBodies);
15186c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
151917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numAttachments = EmailContent.count(
152017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext, Attachment.CONTENT_URI, attachmentSelection, selArgs);
15216c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        assertEquals(2, numAttachments);
15226c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
15236c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        // now delete the other message
15246c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        uri = ContentUris.withAppendedId(Message.CONTENT_URI, message2Id);
15256c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
15266c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
15276c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        // make sure there are no bodies or attachments
15286c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        numBodies = EmailContent.count(mMockContext, Body.CONTENT_URI, bodySelection, selArgs);
15296c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        assertEquals(0, numBodies);
15306c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
153117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        numAttachments = EmailContent.count(
153217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                mMockContext, Attachment.CONTENT_URI, attachmentSelection, selArgs);
15336c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        assertEquals(0, numAttachments);
15346c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler    }
15357143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler
1536976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank    /**
1537976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank     * Test that our unique file name algorithm works as expected.  Since this test requires an
1538976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank     * SD card, we check the environment first, and return immediately if none is mounted.
1539976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank     * @throws IOException
1540976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank     */
1541976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank    public void testCreateUniqueFile() throws IOException {
1542976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        // Delete existing files, if they exist
1543976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        if (!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
1544976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            return;
1545976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        }
1546976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        try {
1547976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            String fileName = "A11achm3n1.doc";
1548976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            File uniqueFile = Attachment.createUniqueFile(fileName);
1549976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            assertEquals(fileName, uniqueFile.getName());
1550976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            if (uniqueFile.createNewFile()) {
1551976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                uniqueFile = Attachment.createUniqueFile(fileName);
1552976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                assertEquals("A11achm3n1-2.doc", uniqueFile.getName());
1553976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                if (uniqueFile.createNewFile()) {
1554976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                    uniqueFile = Attachment.createUniqueFile(fileName);
1555976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                    assertEquals("A11achm3n1-3.doc", uniqueFile.getName());
1556976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                }
155717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            }
1558976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            fileName = "A11achm3n1";
1559976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            uniqueFile = Attachment.createUniqueFile(fileName);
1560976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            assertEquals(fileName, uniqueFile.getName());
1561976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            if (uniqueFile.createNewFile()) {
1562976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                uniqueFile = Attachment.createUniqueFile(fileName);
1563976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                assertEquals("A11achm3n1-2", uniqueFile.getName());
1564976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            }
1565976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        } finally {
1566976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            File directory = Environment.getExternalStorageDirectory();
156717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // These are the files that should be created earlier in the test.
156817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            // Make sure
1569976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            // they are deleted for the next go-around
1570976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            String[] fileNames = new String[] {"A11achm3n1.doc", "A11achm3n1-2.doc", "A11achm3n1"};
1571976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            int length = fileNames.length;
1572976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            for (int i = 0; i < length; i++) {
1573976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                File file = new File(directory, fileNames[i]);
1574976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                if (file.exists()) {
1575976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                    file.delete();
1576976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank                }
1577976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank            }
1578976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank        }
1579976f92908dd2f69f21f62690632ff24b08d9f5d3Marc Blank    }
15804119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler
15814119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler    /**
15824119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler     * Test retrieving attachments by message ID (using EmailContent.Attachment.MESSAGE_ID_URI)
15834119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler     */
15844119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler    public void testGetAttachmentByMessageIdUri() {
15854119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler
158617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Note, we don't strictly need accounts, mailboxes or messages to run
158717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // this test.
15884119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        Attachment a1 = ProviderTestUtils.setupAttachment(1, "a1", 100, true, mMockContext);
15894119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        Attachment a2 = ProviderTestUtils.setupAttachment(1, "a2", 200, true, mMockContext);
1590758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        ProviderTestUtils.setupAttachment(2, "a3", 300, true, mMockContext);
1591758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        ProviderTestUtils.setupAttachment(2, "a4", 400, true, mMockContext);
15924119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler
15934119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        // Now ask for the attachments of message id=1
159417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Note: Using the "sort by size" trick to bring them back in expected
159517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // order
15964119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        Uri uri = ContentUris.withAppendedId(Attachment.MESSAGE_ID_URI, 1);
159717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Cursor c = mMockContext.getContentResolver()
1598845db9711917ad8936926fc7a1a179fade8e00d5Tony Mantler                .query(uri, Attachment.CONTENT_PROJECTION, null, null, AttachmentColumns.SIZE);
15994119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        assertEquals(2, c.getCount());
16004119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler
16014119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        try {
16024119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            c.moveToFirst();
16032f288864b621cfb5aee44eda27df463460d33dd3Tony Mantler            Attachment a1Get = EmailContent.getContent(mMockContext, c, Attachment.class);
16044119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            ProviderTestUtils.assertAttachmentEqual("getAttachByUri-1", a1, a1Get);
16054119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            c.moveToNext();
16062f288864b621cfb5aee44eda27df463460d33dd3Tony Mantler            Attachment a2Get = EmailContent.getContent(mMockContext, c, Attachment.class);
16074119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            ProviderTestUtils.assertAttachmentEqual("getAttachByUri-2", a2, a2Get);
16084119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        } finally {
16094119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler            c.close();
16104119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler        }
16114119218e2fd64341ac946fb8f2cbdb796a444cb8Andrew Stadler    }
161254c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
161354c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler    /**
16146c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler     * Test deleting attachments by message ID (using EmailContent.Attachment.MESSAGE_ID_URI)
16156c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler     */
16166c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler    public void testDeleteAttachmentByMessageIdUri() {
16176c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        ContentResolver mockResolver = mMockContext.getContentResolver();
16186c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
161917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Note, we don't strictly need accounts, mailboxes or messages to run
162017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // this test.
16216c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        ProviderTestUtils.setupAttachment(1, "a1", 100, true, mMockContext);
16226c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        ProviderTestUtils.setupAttachment(1, "a2", 200, true, mMockContext);
16236c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        Attachment a3 = ProviderTestUtils.setupAttachment(2, "a3", 300, true, mMockContext);
16246c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        Attachment a4 = ProviderTestUtils.setupAttachment(2, "a4", 400, true, mMockContext);
16256c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
16266c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        // Delete all attachments for message id=1
16276c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        Uri uri = ContentUris.withAppendedId(Attachment.MESSAGE_ID_URI, 1);
16286c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        mockResolver.delete(uri, null, null);
16296c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
163017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Read back all attachments and confirm that we have the expected
163117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // remaining attachments
163217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // (the attachments that are set for message id=2). Note order-by size
163317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // to simplify test.
163417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Cursor c = mockResolver.query(
1635845db9711917ad8936926fc7a1a179fade8e00d5Tony Mantler                Attachment.CONTENT_URI, Attachment.CONTENT_PROJECTION, null, null,
1636845db9711917ad8936926fc7a1a179fade8e00d5Tony Mantler                AttachmentColumns.SIZE);
16376c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        assertEquals(2, c.getCount());
16386c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
16396c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        try {
16406c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler            c.moveToFirst();
16412f288864b621cfb5aee44eda27df463460d33dd3Tony Mantler            Attachment a3Get = EmailContent.getContent(mMockContext, c, Attachment.class);
16426c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler            ProviderTestUtils.assertAttachmentEqual("getAttachByUri-3", a3, a3Get);
16436c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler            c.moveToNext();
16442f288864b621cfb5aee44eda27df463460d33dd3Tony Mantler            Attachment a4Get = EmailContent.getContent(mMockContext, c, Attachment.class);
16456c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler            ProviderTestUtils.assertAttachmentEqual("getAttachByUri-4", a4, a4Get);
16466c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        } finally {
16476c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler            c.close();
16486c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler        }
16496c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler    }
16506c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler
165152e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo    @SmallTest
165263537746479f4b65517bd217c1a5f76d697367ebMarc Blank    public void testGetDefaultAccountNoneExplicitlySet() {
165363537746479f4b65517bd217c1a5f76d697367ebMarc Blank        Account account1 = ProviderTestUtils.setupAccount("account-default-1", false, mMockContext);
165463537746479f4b65517bd217c1a5f76d697367ebMarc Blank        account1.save(mMockContext);
165563537746479f4b65517bd217c1a5f76d697367ebMarc Blank
165663537746479f4b65517bd217c1a5f76d697367ebMarc Blank        // We should find account1 as default
165717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        long defaultAccountId = Account.getDefaultAccountId(mMockContext, Account.NO_ACCOUNT);
165863537746479f4b65517bd217c1a5f76d697367ebMarc Blank        assertEquals(defaultAccountId, account1.mId);
165963537746479f4b65517bd217c1a5f76d697367ebMarc Blank
166052e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo        Account account2 = ProviderTestUtils.setupAccount("account-default-2", false, mMockContext);
166163537746479f4b65517bd217c1a5f76d697367ebMarc Blank        account2.save(mMockContext);
166263537746479f4b65517bd217c1a5f76d697367ebMarc Blank
166352e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo        Account account3 = ProviderTestUtils.setupAccount("account-default-3", false, mMockContext);
166452e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo        account3.save(mMockContext);
166552e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo
166617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // We should find the earliest one as the default, so that it can be
166717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // consistent on
166852e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo        // repeated calls.
166917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        defaultAccountId = Account.getDefaultAccountId(mMockContext, Account.NO_ACCOUNT);
167052e66110d8934cf0a3dd632101a56d4ef45b549aBen Komalo        assertTrue(defaultAccountId == account1.mId);
167163537746479f4b65517bd217c1a5f76d697367ebMarc Blank    }
167263537746479f4b65517bd217c1a5f76d697367ebMarc Blank
16736c21942ec45f561d711b3d74ecca8e62afb735c4Andrew Stadler    /**
167417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie     * Tests of default account behavior. Note that default account behavior is handled differently
167517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie     * now. If there is no last used account, the first account found by our account query is the
167617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie     * default. If there is a last used account, the last used account is our default.
1677758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank     *
167854c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler     * 1.  Simple set/get
167954c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler     * 2.  Moving default between 3 accounts
168054c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler     * 3.  Delete default, make sure another becomes default
168154c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler     */
168217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    public void testGetDefaultAccountWithLastUsedAccount() {
168317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        long lastUsedAccountId = Account.NO_ACCOUNT;
168417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie
168554c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        // There should be no default account if there are no accounts
168617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        long defaultAccountId = Account.getDefaultAccountId(mMockContext, lastUsedAccountId);
1687acd985efb4b29da28953308ba3d5c8f43b760379Ben Komalo        assertEquals(Account.NO_ACCOUNT, defaultAccountId);
168854c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
1689acd985efb4b29da28953308ba3d5c8f43b760379Ben Komalo        Account account1 = ProviderTestUtils.setupAccount("account-default-1", false, mMockContext);
1690acd985efb4b29da28953308ba3d5c8f43b760379Ben Komalo        account1.save(mMockContext);
169154c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        long account1Id = account1.mId;
1692acd985efb4b29da28953308ba3d5c8f43b760379Ben Komalo        Account account2 = ProviderTestUtils.setupAccount("account-default-2", false, mMockContext);
1693acd985efb4b29da28953308ba3d5c8f43b760379Ben Komalo        account2.save(mMockContext);
169454c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        long account2Id = account2.mId;
1695acd985efb4b29da28953308ba3d5c8f43b760379Ben Komalo        Account account3 = ProviderTestUtils.setupAccount("account-default-3", false, mMockContext);
1696acd985efb4b29da28953308ba3d5c8f43b760379Ben Komalo        account3.save(mMockContext);
169754c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        long account3Id = account3.mId;
169854c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
169917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // With three accounts, but none marked default, confirm that the first
170017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // one is the default.
170117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        defaultAccountId = Account.getDefaultAccountId(mMockContext, lastUsedAccountId);
1702acd985efb4b29da28953308ba3d5c8f43b760379Ben Komalo        assertTrue(defaultAccountId == account1Id);
170354c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
170417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // updating lastUsedAccountId locally instead of updating through
170517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Preferences
170617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        lastUsedAccountId = defaultAccountId;
170717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        defaultAccountId = Account.getDefaultAccountId(mMockContext, lastUsedAccountId);
17089e2c6bd5f21f2d19eef7ebfe30e6fdf94ede0857Andrew Stadler        assertEquals(account1Id, defaultAccountId);
170954c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
171017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // updating lastUsedAccountId locally instead of updating through
171117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Preferences
171217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        lastUsedAccountId = account2Id;
171317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        defaultAccountId = Account.getDefaultAccountId(mMockContext, lastUsedAccountId);
1714531ae9d25f4310e4774e136ed74ed9749f940861Marc Blank        assertEquals(account2Id, defaultAccountId);
1715531ae9d25f4310e4774e136ed74ed9749f940861Marc Blank
171617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // updating lastUsedAccountId locally instead of updating through
171717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Preferences
171817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        lastUsedAccountId = account3Id;
171917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        defaultAccountId = Account.getDefaultAccountId(mMockContext, lastUsedAccountId);
1720531ae9d25f4310e4774e136ed74ed9749f940861Marc Blank        assertEquals(account3Id, defaultAccountId);
172154c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
172254c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        // Now delete a non-default account and confirm no change
172354c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        Uri uri = ContentUris.withAppendedId(Account.CONTENT_URI, account1Id);
172454c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
172554c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
172617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        defaultAccountId = Account.getDefaultAccountId(mMockContext, lastUsedAccountId);
1727531ae9d25f4310e4774e136ed74ed9749f940861Marc Blank        assertEquals(account3Id, defaultAccountId);
172854c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
172917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Now confirm deleting the default account and it switches to another
173017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // one
173154c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        uri = ContentUris.withAppendedId(Account.CONTENT_URI, account3Id);
173254c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
173354c1f2bf9a6574240b7c9af253f83a2b566442abAndrew Stadler
173417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        defaultAccountId = Account.getDefaultAccountId(mMockContext, lastUsedAccountId);
1735531ae9d25f4310e4774e136ed74ed9749f940861Marc Blank        assertEquals(account2Id, defaultAccountId);
1736758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
173717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // updating lastUsedAccountId locally instead of updating through
173817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Preferences
173917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        lastUsedAccountId = defaultAccountId;
174017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie
174117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Now delete the final account and confirm there are no default
174217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // accounts again
17439e2c6bd5f21f2d19eef7ebfe30e6fdf94ede0857Andrew Stadler        uri = ContentUris.withAppendedId(Account.CONTENT_URI, account2Id);
17449e2c6bd5f21f2d19eef7ebfe30e6fdf94ede0857Andrew Stadler        mMockContext.getContentResolver().delete(uri, null, null);
17459e2c6bd5f21f2d19eef7ebfe30e6fdf94ede0857Andrew Stadler
174617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        defaultAccountId = Account.getDefaultAccountId(mMockContext, lastUsedAccountId);
174717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(Account.NO_ACCOUNT, defaultAccountId);
17489e2c6bd5f21f2d19eef7ebfe30e6fdf94ede0857Andrew Stadler    }
1749758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
175017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    public static Message setupUnreadMessage(String name,
175117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            long accountId,
175217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            long mailboxId,
175317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            boolean addBody,
175417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            boolean saveIt,
175517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            Context context) {
1756758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        Message msg =
175717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                ProviderTestUtils.setupMessage(name, accountId, mailboxId, addBody, false, context);
1758758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        msg.mFlagRead = false;
1759758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        if (saveIt) {
1760758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank            msg.save(context);
1761758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        }
1762758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        return msg;
1763758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank    }
1764758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
1765758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank    public void testUnreadCountTriggers() {
1766758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        // Start with one account and three mailboxes
1767758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        Account account = ProviderTestUtils.setupAccount("triggers", true, mMockContext);
1768758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        Mailbox boxA = ProviderTestUtils.setupMailbox("boxA", account.mId, true, mMockContext);
1769758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        Mailbox boxB = ProviderTestUtils.setupMailbox("boxB", account.mId, true, mMockContext);
1770758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        Mailbox boxC = ProviderTestUtils.setupMailbox("boxC", account.mId, true, mMockContext);
1771758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
1772758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        // Make sure there are no unreads
1773758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(0, getUnreadCount(boxA.mId));
1774758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(0, getUnreadCount(boxB.mId));
1775758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(0, getUnreadCount(boxC.mId));
1776758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
1777758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        // Create 4 unread messages (only 3 named) in boxA
177817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message1 =
177917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                setupUnreadMessage("message1", account.mId, boxA.mId, false, true, mMockContext);
178017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message2 =
178117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                setupUnreadMessage("message2", account.mId, boxA.mId, false, true, mMockContext);
178217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Message message3 =
178317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                setupUnreadMessage("message3", account.mId, boxA.mId, false, true, mMockContext);
1784758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        setupUnreadMessage("message4", account.mId, boxC.mId, false, true, mMockContext);
1785758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
1786758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        // Make sure the unreads are where we expect them
1787758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(3, getUnreadCount(boxA.mId));
1788758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(0, getUnreadCount(boxB.mId));
1789758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxC.mId));
1790758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
179117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // After deleting message 1, the count in box A should be decremented
179217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // (to 2)
1793758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        ContentResolver cr = mMockContext.getContentResolver();
1794758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        Uri uri = ContentUris.withAppendedId(Message.CONTENT_URI, message1.mId);
1795758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cr.delete(uri, null, null);
1796758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(2, getUnreadCount(boxA.mId));
1797758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(0, getUnreadCount(boxB.mId));
1798758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxC.mId));
1799758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
1800758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        // Move message 2 to box B, leaving 1 in box A and 1 in box B
1801758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        message2.mMailboxKey = boxB.mId;
1802758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        ContentValues cv = new ContentValues();
1803758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cv.put(MessageColumns.MAILBOX_KEY, boxB.mId);
1804758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cr.update(ContentUris.withAppendedId(Message.CONTENT_URI, message2.mId), cv, null, null);
1805758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxA.mId));
1806758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxB.mId));
1807758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxC.mId));
1808758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
1809758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        // Mark message 3 (from box A) read, leaving 0 in box A
1810758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cv.clear();
1811758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cv.put(MessageColumns.FLAG_READ, 1);
1812758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cr.update(ContentUris.withAppendedId(Message.CONTENT_URI, message3.mId), cv, null, null);
1813758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(0, getUnreadCount(boxA.mId));
1814758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxB.mId));
1815758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxC.mId));
1816758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
1817758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        // Move message 3 to box C; should be no change (it's read)
1818758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        message3.mMailboxKey = boxC.mId;
1819758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cv.clear();
1820758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cv.put(MessageColumns.MAILBOX_KEY, boxC.mId);
1821758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cr.update(ContentUris.withAppendedId(Message.CONTENT_URI, message3.mId), cv, null, null);
1822758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(0, getUnreadCount(boxA.mId));
1823758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxB.mId));
1824758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxC.mId));
1825758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank
182617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Mark message 3 unread; it's now in box C, so that box's count should
182717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // go up to 3
1828758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cv.clear();
1829758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cv.put(MessageColumns.FLAG_READ, 0);
1830758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        cr.update(ContentUris.withAppendedId(Message.CONTENT_URI, message3.mId), cv, null, null);
1831758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(0, getUnreadCount(boxA.mId));
1832758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(1, getUnreadCount(boxB.mId));
1833758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank        assertEquals(2, getUnreadCount(boxC.mId));
1834758a532fce2f672673d38b2daa5f67eb757b118bMarc Blank    }
18359627d014e16235eadf981b9165807dc72a14a383Mihai Preda
18369627d014e16235eadf981b9165807dc72a14a383Mihai Preda    /**
18379627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * Test for EmailProvider.createIndex().
18389627d014e16235eadf981b9165807dc72a14a383Mihai Preda     * Check that it returns exacly the same string as the one used previously for index creation.
18399627d014e16235eadf981b9165807dc72a14a383Mihai Preda     */
18409627d014e16235eadf981b9165807dc72a14a383Mihai Preda    public void testCreateIndex() {
184117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String oldStr = "create index message_" + MessageColumns.TIMESTAMP + " on "
184217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                + Message.TABLE_NAME + " (" + MessageColumns.TIMESTAMP + ");";
184317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        String newStr = DBHelper.createIndex(Message.TABLE_NAME, MessageColumns.TIMESTAMP);
18449627d014e16235eadf981b9165807dc72a14a383Mihai Preda        assertEquals(newStr, oldStr);
18459627d014e16235eadf981b9165807dc72a14a383Mihai Preda    }
1846c0c9c33322deecace00a32766e0a1b355aad4b31Marc Blank
18470e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank    public void testDatabaseCorruptionRecovery() {
18480e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        final ContentResolver resolver = mMockContext.getContentResolver();
18490e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        final Context context = mMockContext;
18500e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
18510e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Create account and two mailboxes
18520e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        Account acct = ProviderTestUtils.setupAccount("acct1", true, context);
18530e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", acct.mId, true, context);
18540e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
18550e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Create 4 messages in box1 with bodies
18560e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        ProviderTestUtils.setupMessage("message1", acct.mId, box1.mId, true, true, context);
18570e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        ProviderTestUtils.setupMessage("message2", acct.mId, box1.mId, true, true, context);
18580e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        ProviderTestUtils.setupMessage("message3", acct.mId, box1.mId, true, true, context);
18590e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        ProviderTestUtils.setupMessage("message4", acct.mId, box1.mId, true, true, context);
18600e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
18610e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Confirm there are four messages
18620e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        int count = EmailContent.count(mMockContext, Message.CONTENT_URI, null, null);
18630e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertEquals(4, count);
18640e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Confirm there are four bodies
18650e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        count = EmailContent.count(mMockContext, Body.CONTENT_URI, null, null);
18660e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertEquals(4, count);
18670e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
18680e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Find the EmailProvider.db file
18690e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        File dbFile = mMockContext.getDatabasePath(EmailProvider.DATABASE_NAME);
187017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // The EmailProvider.db database should exist (the provider creates it
187117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // automatically)
18720e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile != null);
18730e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile.exists());
18740e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Delete it, and confirm it is gone
18750e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile.delete());
18760e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertFalse(dbFile.exists());
18770e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
18780e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Find the EmailProviderBody.db file
18790e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        dbFile = mMockContext.getDatabasePath(EmailProvider.BODY_DATABASE_NAME);
18800e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // The EmailProviderBody.db database should still exist
18810e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile != null);
18820e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile.exists());
18830e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
18840e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // URI to uncache the databases
188517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // This simulates the Provider starting up again (otherwise, it will
188617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // still be pointing to
18870e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // the already opened files)
188817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Note that we only have access to the EmailProvider via the
188917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // ContentResolver; therefore,
18900e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // we cannot directly call into the provider and use a URI for this
18910e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        resolver.update(EmailProvider.INTEGRITY_CHECK_URI, null, null, null);
18920e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
189317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // TODO We should check for the deletion of attachment files once this
189417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // is implemented in
18950e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // the provider
1896bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki
18970e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Explanation for what happens below...
189817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // The next time the database is created by the provider, it will notice
189917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // that there's
190017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // already a EmailProviderBody.db file. In this case, it will delete
190117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // that database to
19020e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // ensure that both are in sync (and empty)
19030e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19040e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Confirm there are no bodies
19050e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        count = EmailContent.count(mMockContext, Body.CONTENT_URI, null, null);
19060e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertEquals(0, count);
19070e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19080e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Confirm there are no messages
19090e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        count = EmailContent.count(mMockContext, Message.CONTENT_URI, null, null);
19100e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertEquals(0, count);
19110e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank    }
19120e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19130e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank    public void testBodyDatabaseCorruptionRecovery() {
19140e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        final ContentResolver resolver = mMockContext.getContentResolver();
19150e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        final Context context = mMockContext;
19160e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19170e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Create account and two mailboxes
19180e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        Account acct = ProviderTestUtils.setupAccount("acct1", true, context);
19190e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        Mailbox box1 = ProviderTestUtils.setupMailbox("box1", acct.mId, true, context);
19200e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19210e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Create 4 messages in box1 with bodies
19220e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        ProviderTestUtils.setupMessage("message1", acct.mId, box1.mId, true, true, context);
19230e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        ProviderTestUtils.setupMessage("message2", acct.mId, box1.mId, true, true, context);
19240e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        ProviderTestUtils.setupMessage("message3", acct.mId, box1.mId, true, true, context);
19250e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        ProviderTestUtils.setupMessage("message4", acct.mId, box1.mId, true, true, context);
19260e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19270e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Confirm there are four messages
19280e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        int count = EmailContent.count(mMockContext, Message.CONTENT_URI, null, null);
19290e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertEquals(4, count);
19300e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Confirm there are four bodies
19310e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        count = EmailContent.count(mMockContext, Body.CONTENT_URI, null, null);
19320e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertEquals(4, count);
19330e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19340e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Find the EmailProviderBody.db file
19350e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        File dbFile = mMockContext.getDatabasePath(EmailProvider.BODY_DATABASE_NAME);
193617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // The EmailProviderBody.db database should exist (the provider creates
193717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // it automatically)
19380e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile != null);
19390e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile.exists());
19400e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Delete it, and confirm it is gone
19410e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile.delete());
19420e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertFalse(dbFile.exists());
19430e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19440e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Find the EmailProvider.db file
19450e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        dbFile = mMockContext.getDatabasePath(EmailProvider.DATABASE_NAME);
19460e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // The EmailProviderBody.db database should still exist
19470e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile != null);
19480e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertTrue(dbFile.exists());
19490e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19500e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // URI to uncache the databases
195117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // This simulates the Provider starting up again (otherwise, it will
195217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // still be pointing to
19530e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // the already opened files)
195417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // Note that we only have access to the EmailProvider via the
195517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // ContentResolver; therefore,
19560e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // we cannot directly call into the provider and use a URI for this
19570e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        resolver.update(EmailProvider.INTEGRITY_CHECK_URI, null, null, null);
19580e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
195917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // TODO We should check for the deletion of attachment files once this
196017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // is implemented in
19610e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // the provider
19620e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19630e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Explanation for what happens below...
196417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // The next time the body database is created by the provider, it will
196517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // notice that there's
196617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // already a populated EmailProvider.db file. In this case, it will
196717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // delete that database to
19680e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // ensure that both are in sync (and empty)
19690e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19700e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Confirm there are no messages
19710e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        count = EmailContent.count(mMockContext, Message.CONTENT_URI, null, null);
19720e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertEquals(0, count);
19730e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank
19740e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        // Confirm there are no bodies
19750e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        count = EmailContent.count(mMockContext, Body.CONTENT_URI, null, null);
19760e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank        assertEquals(0, count);
19770e1595c177e40428b267a8696dfc05d015ce6a2fMarc Blank    }
1978694257cb7e7a191b3314cb209b6c38a8a07583efMarc Blank
1979bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki    public void testAccountIsSecurityHold() {
1980bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        final Context context = mMockContext;
1981bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        Account acct1 = ProviderTestUtils.setupAccount("acct1", true, context);
1982bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki
1983bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        Account acct2 = ProviderTestUtils.setupAccount("acct2", false, context);
1984bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        acct2.mFlags |= Account.FLAGS_SECURITY_HOLD;
1985bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        acct2.save(context);
1986bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki
1987bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        assertFalse(Account.isSecurityHold(context, acct1.mId));
1988bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        assertTrue(Account.isSecurityHold(context, acct2.mId));
198917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertFalse(Account.isSecurityHold(context, 9999999)); // No such
199017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                                                               // account
199117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    }
1992bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki
1993bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki    public void testClearAccountHoldFlags() {
1994bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        Account a1 = ProviderTestUtils.setupAccount("holdflag-1", false, mMockContext);
199517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        a1.mFlags = Account.FLAGS_SUPPORTS_SEARCH;
19966e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        a1.mPolicy = new Policy();
1997bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        a1.save(mMockContext);
1998bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        Account a2 = ProviderTestUtils.setupAccount("holdflag-2", false, mMockContext);
199917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        a2.mFlags = Account.FLAGS_SUPPORTS_SMART_FORWARD | Account.FLAGS_SECURITY_HOLD;
20006e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        a2.mPolicy = new Policy();
2001bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        a2.save(mMockContext);
2002bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki
2003bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        // bulk clear
2004bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        Account.clearSecurityHoldOnAllAccounts(mMockContext);
2005bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki
200617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // confirm new values as expected - no hold flags; other flags
200717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // unmolested
2008bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        Account a1a = Account.restoreAccountWithId(mMockContext, a1.mId);
200917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(Account.FLAGS_SUPPORTS_SEARCH, a1a.mFlags);
2010bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki        Account a2a = Account.restoreAccountWithId(mMockContext, a2.mId);
201117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        assertEquals(Account.FLAGS_SUPPORTS_SMART_FORWARD, a2a.mFlags);
2012bcf32320e2600e96c8a9e997a8903bfc3893b35eMakoto Onuki    }
2013574854b528163f3bf1a7cb974aa80082d1768edfMakoto Onuki
2014833fe73b99e62ad9cf6e80c782717c7de1ff12e4Makoto Onuki    private static Message createMessage(Context c, Mailbox b, boolean starred, boolean read) {
201517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        return ProviderTestUtils.setupMessage("1",
201617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                b.mAccountKey,
201717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                b.mId,
201817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
201917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                true,
202017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                c,
202117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                starred,
202217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                read);
20236d8bfa67c438ee18921d13d8bfba876aacaa9ff6Makoto Onuki    }
2024bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank
2025bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank    public void testGetKeyColumnLong() {
2026bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        final Context c = mMockContext;
2027bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        Account a = ProviderTestUtils.setupAccount("acct", true, c);
2028bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        Mailbox b1 = ProviderTestUtils.setupMailbox("box1", a.mId, true, c, Mailbox.TYPE_MAIL);
2029bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        Mailbox b2 = ProviderTestUtils.setupMailbox("box2", a.mId, true, c, Mailbox.TYPE_MAIL);
2030bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        Message m1 = createMessage(c, b1, false, false);
2031bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        Message m2 = createMessage(c, b2, false, false);
2032bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        assertEquals(a.mId, Message.getKeyColumnLong(c, m1.mId, MessageColumns.ACCOUNT_KEY));
2033bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        assertEquals(a.mId, Message.getKeyColumnLong(c, m2.mId, MessageColumns.ACCOUNT_KEY));
2034bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        assertEquals(b1.mId, Message.getKeyColumnLong(c, m1.mId, MessageColumns.MAILBOX_KEY));
2035bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank        assertEquals(b2.mId, Message.getKeyColumnLong(c, m2.mId, MessageColumns.MAILBOX_KEY));
2036bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank    }
2037bca4e6e70b53ca7db0ac14522f0d26a7b465cf24Marc Blank
2038aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki    public void testGetAccountIdForMessageId() {
2039aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        final Context c = mMockContext;
2040aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        Account a1 = ProviderTestUtils.setupAccount("acct1", true, c);
2041aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        Account a2 = ProviderTestUtils.setupAccount("acct2", true, c);
2042aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        Mailbox b1 = ProviderTestUtils.setupMailbox("box1", a1.mId, true, c, Mailbox.TYPE_MAIL);
2043aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        Mailbox b2 = ProviderTestUtils.setupMailbox("box2", a2.mId, true, c, Mailbox.TYPE_MAIL);
2044aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        Message m1 = createMessage(c, b1, false, false);
2045aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        Message m2 = createMessage(c, b2, false, false);
2046aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki
2047aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        assertEquals(a1.mId, Account.getAccountIdForMessageId(c, m1.mId));
2048aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        assertEquals(a2.mId, Account.getAccountIdForMessageId(c, m2.mId));
2049aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki
2050aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        // message desn't exist
2051aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki        assertEquals(-1, Account.getAccountIdForMessageId(c, 12345));
2052aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki    }
2053aef9515ee70f1f0b6cc4fa601078597b55831331Makoto Onuki
2054200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy    public void testGetAccountForMessageId() {
2055200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy        final Context c = mMockContext;
2056200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy        Account a = ProviderTestUtils.setupAccount("acct", true, c);
2057200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy        Message m1 = ProviderTestUtils.setupMessage("1", a.mId, 1, true, true, c, false, false);
2058200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy        Message m2 = ProviderTestUtils.setupMessage("1", a.mId, 2, true, true, c, false, false);
2059200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy        ProviderTestUtils.assertAccountEqual("x", a, Account.getAccountForMessageId(c, m1.mId));
2060200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy        ProviderTestUtils.assertAccountEqual("x", a, Account.getAccountForMessageId(c, m2.mId));
2061200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy    }
2062200c6bd9fa19b78acc2c1664f858521aa9885353Todd Kennedy
2063e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki    public void testGetAccountGetInboxIdTest() {
2064e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki        final Context c = mMockContext;
2065e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki
2066e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki        // Prepare some data with red-herrings.
2067e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki        Account a2 = ProviderTestUtils.setupAccount("acct2", true, c);
2068e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki        Mailbox b2i = ProviderTestUtils.setupMailbox("b2b", a2.mId, true, c, Mailbox.TYPE_INBOX);
2069e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki
2070e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki        assertEquals(b2i.mId, Account.getInboxId(c, a2.mId));
2071d25d87c7ba8bbbbcad771695a7085f227bac8a1bMakoto Onuki
2072d25d87c7ba8bbbbcad771695a7085f227bac8a1bMakoto Onuki        // No account found.
2073d25d87c7ba8bbbbcad771695a7085f227bac8a1bMakoto Onuki        assertEquals(-1, Account.getInboxId(c, 999999));
2074e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki    }
2075e357f5879187124c7af5c2ece5d7d3e4f60f07d2Makoto Onuki
2076261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki    /**
2077261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki     * Check if update to {@link Account#RESET_NEW_MESSAGE_COUNT_URI} resets the new message count.
2078261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki     */
2079261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki    public void testResetNewMessageCount() {
2080261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        final Context c = mMockContext;
2081261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        final ContentResolver cr = c.getContentResolver();
2082261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki
2083261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        // Prepare test data
2084261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        Account a1 = ProviderTestUtils.setupAccount("acct1", false, c);
2085261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a1.mNewMessageCount = 1;
2086261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a1.save(c);
2087261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        Account a2 = ProviderTestUtils.setupAccount("acct2", false, c);
2088261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a2.mNewMessageCount = 2;
2089261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a2.save(c);
2090261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        Account a3 = ProviderTestUtils.setupAccount("acct3", false, c);
2091261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a3.mNewMessageCount = 3;
2092261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a3.save(c);
2093261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        Account a4 = ProviderTestUtils.setupAccount("acct4", false, c);
2094261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a4.mNewMessageCount = 4;
2095261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a4.save(c);
2096261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        Account a5 = ProviderTestUtils.setupAccount("acct5", false, c);
2097261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a5.mNewMessageCount = 5;
2098261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        a5.save(c);
2099261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki
2100261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        // With ID in URI, no selection
210117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        cr.update(ContentUris.withAppendedId(Account.RESET_NEW_MESSAGE_COUNT_URI, a1.mId), null,
210217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                null, null);
2103261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a1.mId).mNewMessageCount);
2104261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(2, Account.restoreAccountWithId(c, a2.mId).mNewMessageCount);
2105261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(3, Account.restoreAccountWithId(c, a3.mId).mNewMessageCount);
2106261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(4, Account.restoreAccountWithId(c, a4.mId).mNewMessageCount);
2107261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(5, Account.restoreAccountWithId(c, a5.mId).mNewMessageCount);
2108261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki
2109261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        // No ID in URI, with selection
211017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        cr.update(Account.RESET_NEW_MESSAGE_COUNT_URI, null, EmailContent.ID_SELECTION,
211117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                new String[] {Long.toString(a2.mId)});
2112261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a1.mId).mNewMessageCount);
2113261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a2.mId).mNewMessageCount);
2114261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(3, Account.restoreAccountWithId(c, a3.mId).mNewMessageCount);
2115261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(4, Account.restoreAccountWithId(c, a4.mId).mNewMessageCount);
2116261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(5, Account.restoreAccountWithId(c, a5.mId).mNewMessageCount);
2117261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki
2118261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        // With ID, with selection
2119261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        cr.update(ContentUris.withAppendedId(Account.RESET_NEW_MESSAGE_COUNT_URI, a3.mId), null,
2120261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki                EmailContent.ID_SELECTION, new String[] {Long.toString(a3.mId)});
2121261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a1.mId).mNewMessageCount);
2122261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a2.mId).mNewMessageCount);
2123261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a3.mId).mNewMessageCount);
2124261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(4, Account.restoreAccountWithId(c, a4.mId).mNewMessageCount);
2125261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(5, Account.restoreAccountWithId(c, a5.mId).mNewMessageCount);
2126261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki
2127261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        // No ID in URI, no selection
2128261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        cr.update(Account.RESET_NEW_MESSAGE_COUNT_URI, null, null, null);
2129261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a1.mId).mNewMessageCount);
2130261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a2.mId).mNewMessageCount);
2131261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a3.mId).mNewMessageCount);
2132261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a4.mId).mNewMessageCount);
2133261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a5.mId).mNewMessageCount);
2134261d6c3f0c97a12256519a2c3b131a56e57ab45fMakoto Onuki    }
2135899c5b866192a4c4a12413446d10e5d98dbf94faMakoto Onuki
21369d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    /**
21379d5aaeacd6b222877f25924818317c9153708261Makoto Onuki     * Check if update on ACCOUNT_ID_ADD_TO_FIELD updates the cache properly.
21389d5aaeacd6b222877f25924818317c9153708261Makoto Onuki     */
21399d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    public void testUpdateCacheAccountIdAddToField() {
21409d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        final Context c = mMockContext;
214117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie
214217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // make sure Account.CONTENT_URI is defined
214317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        EmailContent.init(c);
214417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        Account.initAccount();
214517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie
21469d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        Account a1 = ProviderTestUtils.setupAccount("a1", true, c);
21479d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21489d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        int start = Account.restoreAccountWithId(c, a1.mId).mNewMessageCount;
21499d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21509d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        // +1 to NEW_MESSAGE_COUNT
21519d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        ContentValues cv = new ContentValues();
21529d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        cv.put(EmailContent.FIELD_COLUMN_NAME, AccountColumns.NEW_MESSAGE_COUNT);
21539d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        cv.put(EmailContent.ADD_COLUMN_NAME, 1);
215417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        mProvider.update(ContentUris.withAppendedId(Account.CONTENT_URI, a1.mId), cv, null, null);
21559d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21569d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        // Check
21579d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        assertEquals(start + 1, Account.restoreAccountWithId(c, a1.mId).mNewMessageCount);
21589d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    }
21599d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21609d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    /**
21619d5aaeacd6b222877f25924818317c9153708261Makoto Onuki     * Check if update on ACCOUNT_RESET_NEW_COUNT updates the cache properly.
21629d5aaeacd6b222877f25924818317c9153708261Makoto Onuki     */
21639d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    public void testUpdateCacheAccountResetNewCount() {
21649d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        final Context c = mMockContext;
21659d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        Account a1 = ProviderTestUtils.setupAccount("a1", true, c);
21669d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21679d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        // precondition
21689d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        assertTrue(Account.restoreAccountWithId(c, a1.mId).mNewMessageCount > 0);
21699d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21709d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        // Reset
21719d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        mProvider.update(Account.RESET_NEW_MESSAGE_COUNT_URI, null, null, null);
21729d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21739d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        // Check
21749d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a1.mId).mNewMessageCount);
21759d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    }
21769d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21779d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    /**
21789d5aaeacd6b222877f25924818317c9153708261Makoto Onuki     * Check if update on ACCOUNT_RESET_NEW_COUNT_ID updates the cache properly.
21799d5aaeacd6b222877f25924818317c9153708261Makoto Onuki     */
21809d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    public void testUpdateCacheAccountResetNewCountId() {
21819d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        final Context c = mMockContext;
21829d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        Account a1 = ProviderTestUtils.setupAccount("a1", true, c);
21839d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21849d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        // precondition
21859d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        assertTrue(Account.restoreAccountWithId(c, a1.mId).mNewMessageCount > 0);
21869d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21879d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        // Reset
21889d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        mProvider.update(ContentUris.withAppendedId(Account.RESET_NEW_MESSAGE_COUNT_URI, a1.mId),
21899d5aaeacd6b222877f25924818317c9153708261Makoto Onuki                null, null, null);
21909d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21919d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        // Check
21929d5aaeacd6b222877f25924818317c9153708261Makoto Onuki        assertEquals(0, Account.restoreAccountWithId(c, a1.mId).mNewMessageCount);
21939d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    }
21949d5aaeacd6b222877f25924818317c9153708261Makoto Onuki
21959d5aaeacd6b222877f25924818317c9153708261Makoto Onuki    /**
2196d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank     * Check that we're handling illegal uri's properly (by throwing an exception unless it's a
2197d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank     * query for an id of -1, in which case we return a zero-length cursor)
2198d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank     */
2199d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank    public void testIllegalUri() {
2200d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        final ContentResolver cr = mMockContext.getContentResolver();
2201d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank
2202d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        ContentValues cv = new ContentValues();
2203d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        Uri uri = Uri.parse("content://" + EmailContent.AUTHORITY + "/fooble");
2204d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        try {
2205d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            cr.insert(uri, cv);
2206d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            fail("Insert should have thrown exception");
2207d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        } catch (IllegalArgumentException e) {
2208d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        }
2209d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        try {
2210d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            cr.update(uri, cv, null, null);
2211d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            fail("Update should have thrown exception");
2212d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        } catch (IllegalArgumentException e) {
2213d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        }
2214d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        try {
2215d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            cr.delete(uri, null, null);
2216d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            fail("Delete should have thrown exception");
2217d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        } catch (IllegalArgumentException e) {
2218d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        }
2219d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        try {
2220d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            cr.query(uri, EmailContent.ID_PROJECTION, null, null, null);
2221d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            fail("Query should have thrown exception");
2222d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        } catch (IllegalArgumentException e) {
2223d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        }
2224d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        uri = Uri.parse("content://" + EmailContent.AUTHORITY + "/mailbox/fred");
2225d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        try {
2226d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            cr.query(uri, EmailContent.ID_PROJECTION, null, null, null);
2227d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank            fail("Query should have thrown exception");
2228d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        } catch (IllegalArgumentException e) {
2229d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        }
2230d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        uri = Uri.parse("content://" + EmailContent.AUTHORITY + "/mailbox/-1");
2231d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        Cursor c = cr.query(uri, EmailContent.ID_PROJECTION, null, null, null);
2232d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        assertNotNull(c);
2233d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        assertEquals(0, c.getCount());
2234d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank        c.close();
2235d306ba34387f3a7e77a4b8d98c6ac45cc14b95adMarc Blank    }
223622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
223722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    /**
223822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy     * Verify {@link EmailProvider#recalculateMessageCount(android.database.sqlite.SQLiteDatabase)}
223922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy     */
224022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    public void testRecalculateMessageCounts() {
224122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        final Context c = mMockContext;
224222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
224322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // Create accounts
224422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Account a1 = ProviderTestUtils.setupAccount("holdflag-1", true, c);
224522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Account a2 = ProviderTestUtils.setupAccount("holdflag-2", true, c);
224622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
224722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // Create mailboxes for each account
224822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Mailbox b1 = ProviderTestUtils.setupMailbox("box1", a1.mId, true, c, Mailbox.TYPE_INBOX);
224922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Mailbox b2 = ProviderTestUtils.setupMailbox("box2", a1.mId, true, c, Mailbox.TYPE_OUTBOX);
225022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Mailbox b3 = ProviderTestUtils.setupMailbox("box3", a2.mId, true, c, Mailbox.TYPE_INBOX);
225122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Mailbox b4 = ProviderTestUtils.setupMailbox("box4", a2.mId, true, c, Mailbox.TYPE_OUTBOX);
225222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Mailbox bt = ProviderTestUtils.setupMailbox("boxT", a2.mId, true, c, Mailbox.TYPE_TRASH);
225322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
225422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // Create some messages
225522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // b1 (account 1, inbox): 1 message, including 1 starred
225622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message m11 = createMessage(c, b1, true, false, Message.FLAG_LOADED_COMPLETE);
225722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
225822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // b2 (account 1, outbox): 2 message, including 1 starred
225922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message m21 = createMessage(c, b2, false, false, Message.FLAG_LOADED_COMPLETE);
226022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message m22 = createMessage(c, b2, true, true, Message.FLAG_LOADED_COMPLETE);
226122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
226222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // b3 (account 2, inbox): 3 message, including 1 starred
226322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message m31 = createMessage(c, b3, false, false, Message.FLAG_LOADED_COMPLETE);
226422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message m32 = createMessage(c, b3, false, false, Message.FLAG_LOADED_COMPLETE);
226522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message m33 = createMessage(c, b3, true, true, Message.FLAG_LOADED_COMPLETE);
226622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
226722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // b4 (account 2, outbox) has no messages.
226822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
226922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // bt (account 2, trash) has 3 messages, including 2 starred
227022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message mt1 = createMessage(c, bt, true, false, Message.FLAG_LOADED_COMPLETE);
227122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message mt2 = createMessage(c, bt, true, false, Message.FLAG_LOADED_COMPLETE);
227222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Message mt3 = createMessage(c, bt, false, false, Message.FLAG_LOADED_COMPLETE);
227322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
227422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // Verifiy initial message counts
227522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(1, getMessageCount(b1.mId));
227622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(2, getMessageCount(b2.mId));
227722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(3, getMessageCount(b3.mId));
227822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(0, getMessageCount(b4.mId));
227922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(3, getMessageCount(bt.mId));
228022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
228122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // Whew. The setup is done; now let's actually get to the test
228222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
228322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // First, invalidate the message counts.
228422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        setMinusOneToMessageCounts();
228522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(-1, getMessageCount(b1.mId));
228622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(-1, getMessageCount(b2.mId));
228722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(-1, getMessageCount(b3.mId));
228822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(-1, getMessageCount(b4.mId));
228922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(-1, getMessageCount(bt.mId));
229022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
229122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // Batch update.
229222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        SQLiteDatabase db = getProvider().getDatabase(mMockContext);
229317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        DBHelper.recalculateMessageCount(db);
229422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
229522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // Check message counts are valid again
229622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(1, getMessageCount(b1.mId));
229722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(2, getMessageCount(b2.mId));
229822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(3, getMessageCount(b3.mId));
229922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(0, getMessageCount(b4.mId));
230022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(3, getMessageCount(bt.mId));
230122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    }
230222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
230322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    /** Creates an account */
230422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    private Account createAccount(Context c, String name, HostAuth recvAuth, HostAuth sendAuth) {
230522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Account account = ProviderTestUtils.setupAccount(name, false, c);
230622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        if (recvAuth != null) {
230722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy            account.mHostAuthKeyRecv = recvAuth.mId;
230822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy            if (sendAuth == null) {
230922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy                account.mHostAuthKeySend = recvAuth.mId;
231022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy            }
231122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        }
231222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        if (sendAuth != null) {
231322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy            account.mHostAuthKeySend = sendAuth.mId;
231422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        }
231522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        account.save(c);
231622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        return account;
231722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    }
231822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
231922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    /** Creates a mailbox; redefine as we need version 17 mailbox values */
232017d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    private Mailbox createMailbox(
232117d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            Context c, String displayName, String serverId, long parentKey, long accountId) {
232222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        Mailbox box = new Mailbox();
232322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
232422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mDisplayName = displayName;
232522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mServerId = serverId;
232622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mParentKey = parentKey;
232722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mAccountKey = accountId;
232822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        // Don't care about the fields below ... set them for giggles
232922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mType = Mailbox.TYPE_MAIL;
233022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mDelimiter = '/';
233122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mSyncKey = "sync-key";
233222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mSyncLookback = 2;
2333f5418f1f93b02e7fab9f15eb201800b65510998eMarc Blank        box.mSyncInterval = Account.CHECK_INTERVAL_NEVER;
233422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mSyncTime = 3;
233522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mFlagVisible = true;
233622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.mFlags = 5;
233722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        box.save(c);
233822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        return box;
233922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    }
234022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
234122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    /**
234222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy     * Asserts equality between two mailboxes. We define this as we don't have implementations
234322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy     * for Mailbox#equals().
234422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy     */
234522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    private void assertEquals(Mailbox expected, Mailbox actual) {
234622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        if (expected == null && actual == null) return;
234722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertTrue(expected != null && actual != null);
234822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEqualsExceptServerId(expected, actual, expected.mServerId);
234922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    }
235022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
235122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    /**
235222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy     * Asserts equality between the two mailboxes EXCEPT for the server id. The given server
235322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy     * ID is the expected value.
235422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy     */
235522208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    private void assertEqualsExceptServerId(Mailbox expected, Mailbox actual, String serverId) {
235622208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        if (expected == null && actual == null) return;
235722208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
235822208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertTrue(expected != null && actual != null);
235922208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(expected.mDisplayName, actual.mDisplayName);
236022208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(serverId, actual.mServerId);
236122208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(expected.mParentKey, actual.mParentKey);
236222208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy        assertEquals(expected.mAccountKey, actual.mAccountKey);
236322208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy    }
236422208771b7b39c5d131372ba6bc45ab23cc22232Todd Kennedy
2365f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank    /**
2366f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank     * Determine whether a list of AccountManager accounts includes a given EmailProvider account
2367f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank     * @param amAccountList a list of AccountManager accounts
2368f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank     * @param account an EmailProvider account
2369f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank     * @param context the caller's context (our test provider's context)
2370f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank     * @return whether or not the EmailProvider account is represented in AccountManager
2371f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank     */
237217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie    private boolean amAccountListHasAccount(
237317d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie            android.accounts.Account[] amAccountList, Account account, Context context) {
237432bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo        String email = account.mEmailAddress;
237517d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        for (android.accounts.Account amAccount : amAccountList) {
237632bed4bb8e23d7322ab338773d135845f392d3cfBen Komalo            if (amAccount.name.equals(email)) {
2377f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank                return true;
2378f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank            }
2379f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank        }
2380f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank        return false;
2381f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank    }
2382f3ff0ba91076ef1fb087fc30fe65d9504011c2b3Marc Blank
23836e418aa41a17136be0dddb816d843428a0a1e722Marc Blank    /** Creates a mailbox; redefine as we need version 17 mailbox values */
23846e418aa41a17136be0dddb816d843428a0a1e722Marc Blank    private Mailbox createTypeMailbox(Context c, long accountId, int type) {
23856e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        Mailbox box = new Mailbox();
23866e418aa41a17136be0dddb816d843428a0a1e722Marc Blank
23876e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        box.mDisplayName = "foo";
23886e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        box.mServerId = "1:1";
23896e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        box.mParentKey = 0;
23906e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        box.mAccountKey = accountId;
23916e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        // Don't care about the fields below ... set them for giggles
23926e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        box.mType = type;
23936e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        box.save(c);
23946e418aa41a17136be0dddb816d843428a0a1e722Marc Blank        return box;
23956e418aa41a17136be0dddb816d843428a0a1e722Marc Blank    }
23966e418aa41a17136be0dddb816d843428a0a1e722Marc Blank
23972bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank    public void testCleanupOrphans() {
23982bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        EmailProvider ep = getProvider();
23992bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        SQLiteDatabase db = ep.getDatabase(mMockContext);
24002bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank
24012bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Account a = ProviderTestUtils.setupAccount("account1", true, mMockContext);
24022bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        // Mailbox a1 and a3 won't have a valid account
24032bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Mailbox a1 = createTypeMailbox(mMockContext, -1, Mailbox.TYPE_INBOX);
24042bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Mailbox a2 = createTypeMailbox(mMockContext, a.mId, Mailbox.TYPE_MAIL);
24052bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Mailbox a3 = createTypeMailbox(mMockContext, -1, Mailbox.TYPE_DRAFTS);
24062bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Mailbox a4 = createTypeMailbox(mMockContext, a.mId, Mailbox.TYPE_SENT);
24072bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Mailbox a5 = createTypeMailbox(mMockContext, a.mId, Mailbox.TYPE_TRASH);
24082bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        // Mailbox ax isn't even saved; use an obviously invalid id
24092bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Mailbox ax = new Mailbox();
24102bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        ax.mId = 69105;
24112bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank
24122bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        // Message mt2 is an orphan, as is mt4
24132bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Message m1 = createMessage(mMockContext, a1, true, false, Message.FLAG_LOADED_COMPLETE);
24142bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Message m2 = createMessage(mMockContext, a2, true, false, Message.FLAG_LOADED_COMPLETE);
24152bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Message m3 = createMessage(mMockContext, a3, true, false, Message.FLAG_LOADED_COMPLETE);
24162bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Message m4 = createMessage(mMockContext, a4, true, false, Message.FLAG_LOADED_COMPLETE);
24172bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Message m5 = createMessage(mMockContext, a5, true, false, Message.FLAG_LOADED_COMPLETE);
24182bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Message mx = createMessage(mMockContext, ax, true, false, Message.FLAG_LOADED_COMPLETE);
24192bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank
24202bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        // Two orphan policies
24212bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Policy p1 = new Policy();
24222bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        p1.save(mMockContext);
24232bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        Policy p2 = new Policy();
24242bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        p2.save(mMockContext);
24252bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank
242617d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // We don't want anything cached or the tests below won't work. Note
242717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // that
242817d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // deleteUnlinked is only called by EmailProvider when the caches are
242917d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        // empty
24302bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        ContentCache.invalidateAllCaches();
24312bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        // Delete orphaned mailboxes/messages/policies
243217d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        EmailProvider.deleteUnlinked(db, Mailbox.TABLE_NAME, MailboxColumns.ACCOUNT_KEY,
24333dd85723a1af5537e23e4b05bdc361cce9cd42beTony Mantler                AccountColumns._ID, Account.TABLE_NAME);
243417d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie        EmailProvider.deleteUnlinked(db, Message.TABLE_NAME, MessageColumns.ACCOUNT_KEY,
24353dd85723a1af5537e23e4b05bdc361cce9cd42beTony Mantler                AccountColumns._ID, Account.TABLE_NAME);
24363dd85723a1af5537e23e4b05bdc361cce9cd42beTony Mantler        EmailProvider.deleteUnlinked(db, Policy.TABLE_NAME, PolicyColumns._ID,
243717d3a29c9d8f7a27c463239f190bdcc4e0804527Jerry Xie                AccountColumns.POLICY_KEY, Account.TABLE_NAME);
24382bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank
24392bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        // Make sure the orphaned mailboxes are gone
24402bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNull(Mailbox.restoreMailboxWithId(mMockContext, a1.mId));
24412bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNotNull(Mailbox.restoreMailboxWithId(mMockContext, a2.mId));
24422bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNull(Mailbox.restoreMailboxWithId(mMockContext, a3.mId));
24432bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNotNull(Mailbox.restoreMailboxWithId(mMockContext, a4.mId));
24442bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNotNull(Mailbox.restoreMailboxWithId(mMockContext, a5.mId));
24452bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNull(Mailbox.restoreMailboxWithId(mMockContext, ax.mId));
24462bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank
24472bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        // Make sure orphaned messages are gone
24482bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNull(Message.restoreMessageWithId(mMockContext, m1.mId));
24492bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNotNull(Message.restoreMessageWithId(mMockContext, m2.mId));
24502bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNull(Message.restoreMessageWithId(mMockContext, m3.mId));
24512bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNotNull(Message.restoreMessageWithId(mMockContext, m4.mId));
24522bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNotNull(Message.restoreMessageWithId(mMockContext, m5.mId));
24532bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNull(Message.restoreMessageWithId(mMockContext, mx.mId));
24542bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank
24552bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        // Make sure orphaned policies are gone
24562bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNull(Policy.restorePolicyWithId(mMockContext, p1.mId));
24572bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNull(Policy.restorePolicyWithId(mMockContext, p2.mId));
24582bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        a = Account.restoreAccountWithId(mMockContext, a.mId);
24592bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank        assertNotNull(Policy.restorePolicyWithId(mMockContext, a.mPolicyKey));
24602bdf7ee0f0f4a2b11b5f7c0f8b193080600fefd8Marc Blank    }
24617143d969dde180b7a44bdc42ade6ca2878d0760dAndrew Stadler}
2462