LegacyConversions.java revision dfd53b0e821a6c48f128f62566066876a063fed6
1/*
2 * Copyright (C) 2009 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package com.android.email;
18
19import com.android.email.mail.Address;
20import com.android.email.mail.Body;
21import com.android.email.mail.Flag;
22import com.android.email.mail.Folder;
23import com.android.email.mail.Message;
24import com.android.email.mail.MessagingException;
25import com.android.email.mail.Part;
26import com.android.email.mail.Message.RecipientType;
27import com.android.email.mail.internet.MimeBodyPart;
28import com.android.email.mail.internet.MimeHeader;
29import com.android.email.mail.internet.MimeMessage;
30import com.android.email.mail.internet.MimeMultipart;
31import com.android.email.mail.internet.MimeUtility;
32import com.android.email.mail.internet.TextBody;
33import com.android.email.mail.store.LocalStore;
34import com.android.email.provider.AttachmentProvider;
35import com.android.email.provider.EmailContent;
36import com.android.email.provider.EmailContent.Attachment;
37import com.android.email.provider.EmailContent.AttachmentColumns;
38import com.android.email.provider.EmailContent.Mailbox;
39
40import org.apache.commons.io.IOUtils;
41
42import android.content.ContentUris;
43import android.content.ContentValues;
44import android.content.Context;
45import android.database.Cursor;
46import android.net.Uri;
47import android.provider.OpenableColumns;
48import android.util.Log;
49
50import java.io.File;
51import java.io.FileOutputStream;
52import java.io.IOException;
53import java.io.InputStream;
54import java.util.ArrayList;
55import java.util.Date;
56import java.util.HashMap;
57
58public class LegacyConversions {
59
60    /** DO NOT CHECK IN "TRUE" */
61    private static final boolean DEBUG_ATTACHMENTS = false;
62
63    /** Used for mapping folder names to type codes (e.g. inbox, drafts, trash) */
64    private static final HashMap<String, Integer>
65            sServerMailboxNames = new HashMap<String, Integer>();
66
67    /**
68     * Values for HEADER_ANDROID_BODY_QUOTED_PART to tag body parts
69     */
70    /* package */ static final String BODY_QUOTED_PART_REPLY = "quoted-reply";
71    /* package */ static final String BODY_QUOTED_PART_FORWARD = "quoted-forward";
72    /* package */ static final String BODY_QUOTED_PART_INTRO = "quoted-intro";
73
74    /**
75     * Standard columns for querying content providers
76     */
77    private static final String[] ATTACHMENT_META_COLUMNS_PROJECTION = {
78        OpenableColumns.DISPLAY_NAME,
79        OpenableColumns.SIZE
80    };
81    private static final int ATTACHMENT_META_COLUMNS_SIZE = 1;
82
83    /**
84     * Copy field-by-field from a "store" message to a "provider" message
85     * @param message The message we've just downloaded (must be a MimeMessage)
86     * @param localMessage The message we'd like to write into the DB
87     * @result true if dirty (changes were made)
88     */
89    public static boolean updateMessageFields(EmailContent.Message localMessage, Message message,
90                long accountId, long mailboxId) throws MessagingException {
91
92        Address[] from = message.getFrom();
93        Address[] to = message.getRecipients(Message.RecipientType.TO);
94        Address[] cc = message.getRecipients(Message.RecipientType.CC);
95        Address[] bcc = message.getRecipients(Message.RecipientType.BCC);
96        Address[] replyTo = message.getReplyTo();
97        String subject = message.getSubject();
98        Date sentDate = message.getSentDate();
99        Date internalDate = message.getInternalDate();
100
101        if (from != null && from.length > 0) {
102            localMessage.mDisplayName = from[0].toFriendly();
103        }
104        if (sentDate != null) {
105            localMessage.mTimeStamp = sentDate.getTime();
106        }
107        if (subject != null) {
108            localMessage.mSubject = subject;
109        }
110        localMessage.mFlagRead = message.isSet(Flag.SEEN);
111
112        // Keep the message in the "unloaded" state until it has (at least) a display name.
113        // This prevents early flickering of empty messages in POP download.
114        if (localMessage.mFlagLoaded != EmailContent.Message.FLAG_LOADED_COMPLETE) {
115            if (localMessage.mDisplayName == null || "".equals(localMessage.mDisplayName)) {
116                localMessage.mFlagLoaded = EmailContent.Message.FLAG_LOADED_UNLOADED;
117            } else {
118                localMessage.mFlagLoaded = EmailContent.Message.FLAG_LOADED_PARTIAL;
119            }
120        }
121        localMessage.mFlagFavorite = message.isSet(Flag.FLAGGED);
122//        public boolean mFlagAttachment = false;
123//        public int mFlags = 0;
124
125        localMessage.mServerId = message.getUid();
126        if (internalDate != null) {
127            localMessage.mServerTimeStamp = internalDate.getTime();
128        }
129//        public String mClientId;
130
131        // Only replace the local message-id if a new one was found.  This is seen in some ISP's
132        // which may deliver messages w/o a message-id header.
133        String messageId = ((MimeMessage)message).getMessageId();
134        if (messageId != null) {
135            localMessage.mMessageId = messageId;
136        }
137
138//        public long mBodyKey;
139        localMessage.mMailboxKey = mailboxId;
140        localMessage.mAccountKey = accountId;
141
142        if (from != null && from.length > 0) {
143            localMessage.mFrom = Address.pack(from);
144        }
145
146        localMessage.mTo = Address.pack(to);
147        localMessage.mCc = Address.pack(cc);
148        localMessage.mBcc = Address.pack(bcc);
149        localMessage.mReplyTo = Address.pack(replyTo);
150
151//        public String mText;
152//        public String mHtml;
153//        public String mTextReply;
154//        public String mHtmlReply;
155
156//        // Can be used while building messages, but is NOT saved by the Provider
157//        transient public ArrayList<Attachment> mAttachments = null;
158
159        return true;
160    }
161
162    /**
163     * Copy body text (plain and/or HTML) from MimeMessage to provider Message
164     */
165    public static boolean updateBodyFields(EmailContent.Body body,
166            EmailContent.Message localMessage, ArrayList<Part> viewables)
167            throws MessagingException {
168
169        body.mMessageKey = localMessage.mId;
170
171        StringBuffer sbHtml = null;
172        StringBuffer sbText = null;
173        StringBuffer sbHtmlReply = null;
174        StringBuffer sbTextReply = null;
175        StringBuffer sbIntroText = null;
176
177        for (Part viewable : viewables) {
178            String text = MimeUtility.getTextFromPart(viewable);
179            String[] replyTags = viewable.getHeader(MimeHeader.HEADER_ANDROID_BODY_QUOTED_PART);
180            String replyTag = null;
181            if (replyTags != null && replyTags.length > 0) {
182                replyTag = replyTags[0];
183            }
184            // Deploy text as marked by the various tags
185            boolean isHtml = "text/html".equalsIgnoreCase(viewable.getMimeType());
186
187            if (replyTag != null) {
188                boolean isQuotedReply = BODY_QUOTED_PART_REPLY.equalsIgnoreCase(replyTag);
189                boolean isQuotedForward = BODY_QUOTED_PART_FORWARD.equalsIgnoreCase(replyTag);
190                boolean isQuotedIntro = BODY_QUOTED_PART_INTRO.equalsIgnoreCase(replyTag);
191
192                if (isQuotedReply || isQuotedForward) {
193                    if (isHtml) {
194                        sbHtmlReply = appendTextPart(sbHtmlReply, text);
195                    } else {
196                        sbTextReply = appendTextPart(sbTextReply, text);
197                    }
198                    // Set message flags as well
199                    localMessage.mFlags &= ~EmailContent.Message.FLAG_TYPE_MASK;
200                    localMessage.mFlags |= isQuotedReply
201                            ? EmailContent.Message.FLAG_TYPE_REPLY
202                            : EmailContent.Message.FLAG_TYPE_FORWARD;
203                    continue;
204                }
205                if (isQuotedIntro) {
206                    sbIntroText = appendTextPart(sbIntroText, text);
207                    continue;
208                }
209            }
210
211            // Most of the time, just process regular body parts
212            if (isHtml) {
213                sbHtml = appendTextPart(sbHtml, text);
214            } else {
215                sbText = appendTextPart(sbText, text);
216            }
217        }
218
219        // write the combined data to the body part
220        if (sbText != null && sbText.length() != 0) {
221            body.mTextContent = sbText.toString();
222        }
223        if (sbHtml != null && sbHtml.length() != 0) {
224            body.mHtmlContent = sbHtml.toString();
225        }
226        if (sbHtmlReply != null && sbHtmlReply.length() != 0) {
227            body.mHtmlReply = sbHtmlReply.toString();
228        }
229        if (sbTextReply != null && sbTextReply.length() != 0) {
230            body.mTextReply = sbTextReply.toString();
231        }
232        if (sbIntroText != null && sbIntroText.length() != 0) {
233            body.mIntroText = sbIntroText.toString();
234        }
235        return true;
236    }
237
238    /**
239     * Helper function to append text to a StringBuffer, creating it if necessary.
240     * Optimization:  The majority of the time we are *not* appending - we should have a path
241     * that deals with single strings.
242     */
243    private static StringBuffer appendTextPart(StringBuffer sb, String newText) {
244        if (newText == null) {
245            return sb;
246        }
247        else if (sb == null) {
248            sb = new StringBuffer(newText);
249        } else {
250            if (sb.length() > 0) {
251                sb.append('\n');
252            }
253            sb.append(newText);
254        }
255        return sb;
256    }
257
258    /**
259     * Copy attachments from MimeMessage to provider Message.
260     *
261     * @param context a context for file operations
262     * @param localMessage the attachments will be built against this message
263     * @param attachments the attachments to add
264     * @param upgrading if true, we are upgrading a local account - handle attachments differently
265     * @throws IOException
266     */
267    public static void updateAttachments(Context context, EmailContent.Message localMessage,
268            ArrayList<Part> attachments, boolean upgrading) throws MessagingException, IOException {
269        localMessage.mAttachments = null;
270        for (Part attachmentPart : attachments) {
271            addOneAttachment(context, localMessage, attachmentPart, upgrading);
272        }
273    }
274
275    /**
276     * Add a single attachment part to the message
277     *
278     * This will skip adding attachments if they are already found in the attachments table.
279     * The heuristic for this will fail (false-positive) if two identical attachments are
280     * included in a single POP3 message.
281     * TODO: Fix that, by (elsewhere) simulating an mLocation value based on the attachments
282     * position within the list of multipart/mixed elements.  This would make every POP3 attachment
283     * unique, and might also simplify the code (since we could just look at the positions, and
284     * ignore the filename, etc.)
285     *
286     * TODO: Take a closer look at encoding and deal with it if necessary.
287     *
288     * @param context a context for file operations
289     * @param localMessage the attachments will be built against this message
290     * @param part a single attachment part from POP or IMAP
291     * @param upgrading true if upgrading a local account - handle attachments differently
292     * @throws IOException
293     */
294    private static void addOneAttachment(Context context, EmailContent.Message localMessage,
295            Part part, boolean upgrading) throws MessagingException, IOException {
296
297        Attachment localAttachment = new Attachment();
298
299        // Transfer fields from mime format to provider format
300        String contentType = MimeUtility.unfoldAndDecode(part.getContentType());
301        String name = MimeUtility.getHeaderParameter(contentType, "name");
302        if (name == null) {
303            String contentDisposition = MimeUtility.unfoldAndDecode(part.getContentType());
304            name = MimeUtility.getHeaderParameter(contentDisposition, "filename");
305        }
306
307        // Select the URI for the new attachments.  For attachments downloaded by the legacy
308        // IMAP/POP code, this is not determined yet, so is null (it will be rewritten below,
309        // or later, when the actual attachment file is created.)
310        //
311        // When upgrading older local accounts, the URI represents a local asset (e.g. a photo)
312        // so we need to preserve the URI.
313        // TODO This works for outgoing messages, where the URI does not change.  May need
314        // additional logic to handle the case of rewriting URI for received attachments.
315        Uri contentUri = null;
316        String contentUriString = null;
317        if (upgrading) {
318            Body body = part.getBody();
319            if (body instanceof LocalStore.LocalAttachmentBody) {
320                LocalStore.LocalAttachmentBody localBody = (LocalStore.LocalAttachmentBody) body;
321                contentUri = localBody.getContentUri();
322                if (contentUri != null) {
323                    contentUriString = contentUri.toString();
324                }
325            }
326        }
327
328        // Find size, if available, via a number of techniques:
329        long size = 0;
330        if (upgrading) {
331            // If upgrading a legacy account, the size must be recaptured from the data source
332            if (contentUri != null) {
333                Cursor metadataCursor = context.getContentResolver().query(contentUri,
334                        ATTACHMENT_META_COLUMNS_PROJECTION, null, null, null);
335                if (metadataCursor != null) {
336                    try {
337                        if (metadataCursor.moveToFirst()) {
338                            size = metadataCursor.getInt(ATTACHMENT_META_COLUMNS_SIZE);
339                        }
340                    } finally {
341                        metadataCursor.close();
342                    }
343                }
344            }
345            // TODO: a downloaded legacy attachment - see if the above code works
346        } else {
347            // Incoming attachment: Try to pull size from disposition (if not downloaded yet)
348            String disposition = part.getDisposition();
349            if (disposition != null) {
350                String s = MimeUtility.getHeaderParameter(disposition, "size");
351                if (s != null) {
352                    size = Long.parseLong(s);
353                }
354            }
355        }
356
357        // Get partId for unloaded IMAP attachments (if any)
358        // This is only provided (and used) when we have structure but not the actual attachment
359        String[] partIds = part.getHeader(MimeHeader.HEADER_ANDROID_ATTACHMENT_STORE_DATA);
360        String partId = partIds != null ? partIds[0] : null;
361
362        localAttachment.mFileName = MimeUtility.getHeaderParameter(contentType, "name");
363        localAttachment.mMimeType = part.getMimeType();
364        localAttachment.mSize = size;           // May be reset below if file handled
365        localAttachment.mContentId = part.getContentId();
366        localAttachment.mContentUri = contentUriString;
367        localAttachment.mMessageKey = localMessage.mId;
368        localAttachment.mLocation = partId;
369        localAttachment.mEncoding = "B";        // TODO - convert other known encodings
370
371        if (DEBUG_ATTACHMENTS) {
372            Log.d(Email.LOG_TAG, "Add attachment " + localAttachment);
373        }
374
375        // To prevent duplication - do we already have a matching attachment?
376        // The fields we'll check for equality are:
377        //  mFileName, mMimeType, mContentId, mMessageKey, mLocation
378        // NOTE:  This will false-positive if you attach the exact same file, twice, to a POP3
379        // message.  We can live with that - you'll get one of the copies.
380        Uri uri = ContentUris.withAppendedId(Attachment.MESSAGE_ID_URI, localMessage.mId);
381        Cursor cursor = context.getContentResolver().query(uri, Attachment.CONTENT_PROJECTION,
382                null, null, null);
383        boolean attachmentFoundInDb = false;
384        try {
385            while (cursor.moveToNext()) {
386                Attachment dbAttachment = new Attachment().restore(cursor);
387                // We test each of the fields here (instead of in SQL) because they may be
388                // null, or may be strings.
389                if (stringNotEqual(dbAttachment.mFileName, localAttachment.mFileName)) continue;
390                if (stringNotEqual(dbAttachment.mMimeType, localAttachment.mMimeType)) continue;
391                if (stringNotEqual(dbAttachment.mContentId, localAttachment.mContentId)) continue;
392                if (stringNotEqual(dbAttachment.mLocation, localAttachment.mLocation)) continue;
393                // We found a match, so use the existing attachment id, and stop looking/looping
394                attachmentFoundInDb = true;
395                localAttachment.mId = dbAttachment.mId;
396                if (DEBUG_ATTACHMENTS) {
397                    Log.d(Email.LOG_TAG, "Skipped, found db attachment " + dbAttachment);
398                }
399                break;
400            }
401        } finally {
402            cursor.close();
403        }
404
405        // Save the attachment (so far) in order to obtain an id
406        if (!attachmentFoundInDb) {
407            localAttachment.save(context);
408        }
409
410        // If an attachment body was actually provided, we need to write the file now
411        if (!upgrading) {
412            saveAttachmentBody(context, part, localAttachment, localMessage.mAccountKey);
413        }
414
415        if (localMessage.mAttachments == null) {
416            localMessage.mAttachments = new ArrayList<Attachment>();
417        }
418        localMessage.mAttachments.add(localAttachment);
419        localMessage.mFlagAttachment = true;
420    }
421
422    /**
423     * Helper for addOneAttachment that compares two strings, deals with nulls, and treats
424     * nulls and empty strings as equal.
425     */
426    /* package */ static boolean stringNotEqual(String a, String b) {
427        if (a == null && b == null) return false;       // fast exit for two null strings
428        if (a == null) a = "";
429        if (b == null) b = "";
430        return !a.equals(b);
431    }
432
433    /**
434     * Save the body part of a single attachment, to a file in the attachments directory.
435     */
436    public static void saveAttachmentBody(Context context, Part part, Attachment localAttachment,
437            long accountId) throws MessagingException, IOException {
438        if (part.getBody() != null) {
439            long attachmentId = localAttachment.mId;
440
441            InputStream in = part.getBody().getInputStream();
442
443            File saveIn = AttachmentProvider.getAttachmentDirectory(context, accountId);
444            if (!saveIn.exists()) {
445                saveIn.mkdirs();
446            }
447            File saveAs = AttachmentProvider.getAttachmentFilename(context, accountId,
448                    attachmentId);
449            saveAs.createNewFile();
450            FileOutputStream out = new FileOutputStream(saveAs);
451            long copySize = IOUtils.copy(in, out);
452            in.close();
453            out.close();
454
455            // update the attachment with the extra information we now know
456            String contentUriString = AttachmentProvider.getAttachmentUri(
457                    accountId, attachmentId).toString();
458
459            localAttachment.mSize = copySize;
460            localAttachment.mContentUri = contentUriString;
461
462            // update the attachment in the database as well
463            ContentValues cv = new ContentValues();
464            cv.put(AttachmentColumns.SIZE, copySize);
465            cv.put(AttachmentColumns.CONTENT_URI, contentUriString);
466            Uri uri = ContentUris.withAppendedId(Attachment.CONTENT_URI, attachmentId);
467            context.getContentResolver().update(uri, cv, null, null);
468        }
469    }
470
471    /**
472     * Read a complete Provider message into a legacy message (for IMAP upload).  This
473     * is basically the equivalent of LocalFolder.getMessages() + LocalFolder.fetch().
474     */
475    public static Message makeMessage(Context context, EmailContent.Message localMessage)
476            throws MessagingException {
477        MimeMessage message = new MimeMessage();
478
479        // LocalFolder.getMessages() equivalent:  Copy message fields
480        message.setSubject(localMessage.mSubject == null ? "" : localMessage.mSubject);
481        Address[] from = Address.unpack(localMessage.mFrom);
482        if (from.length > 0) {
483            message.setFrom(from[0]);
484        }
485        message.setSentDate(new Date(localMessage.mTimeStamp));
486        message.setUid(localMessage.mServerId);
487        message.setFlag(Flag.DELETED,
488                localMessage.mFlagLoaded == EmailContent.Message.FLAG_LOADED_DELETED);
489        message.setFlag(Flag.SEEN, localMessage.mFlagRead);
490        message.setFlag(Flag.FLAGGED, localMessage.mFlagFavorite);
491//      message.setFlag(Flag.DRAFT, localMessage.mMailboxKey == draftMailboxKey);
492        message.setRecipients(RecipientType.TO, Address.unpack(localMessage.mTo));
493        message.setRecipients(RecipientType.CC, Address.unpack(localMessage.mCc));
494        message.setRecipients(RecipientType.BCC, Address.unpack(localMessage.mBcc));
495        message.setReplyTo(Address.unpack(localMessage.mReplyTo));
496        message.setInternalDate(new Date(localMessage.mServerTimeStamp));
497        message.setMessageId(localMessage.mMessageId);
498
499        // LocalFolder.fetch() equivalent: build body parts
500        message.setHeader(MimeHeader.HEADER_CONTENT_TYPE, "multipart/mixed");
501        MimeMultipart mp = new MimeMultipart();
502        mp.setSubType("mixed");
503        message.setBody(mp);
504
505        try {
506            addTextBodyPart(mp, "text/html", null,
507                    EmailContent.Body.restoreBodyHtmlWithMessageId(context, localMessage.mId));
508        } catch (RuntimeException rte) {
509            Log.d(Email.LOG_TAG, "Exception while reading html body " + rte.toString());
510        }
511
512        try {
513            addTextBodyPart(mp, "text/plain", null,
514                    EmailContent.Body.restoreBodyTextWithMessageId(context, localMessage.mId));
515        } catch (RuntimeException rte) {
516            Log.d(Email.LOG_TAG, "Exception while reading text body " + rte.toString());
517        }
518
519        boolean isReply = (localMessage.mFlags & EmailContent.Message.FLAG_TYPE_REPLY) != 0;
520        boolean isForward = (localMessage.mFlags & EmailContent.Message.FLAG_TYPE_FORWARD) != 0;
521
522        // If there is a quoted part (forwarding or reply), add the intro first, and then the
523        // rest of it.  If it is opened in some other viewer, it will (hopefully) be displayed in
524        // the same order as we've just set up the blocks:  composed text, intro, replied text
525        if (isReply || isForward) {
526            try {
527                addTextBodyPart(mp, "text/plain", BODY_QUOTED_PART_INTRO,
528                        EmailContent.Body.restoreIntroTextWithMessageId(context, localMessage.mId));
529            } catch (RuntimeException rte) {
530                Log.d(Email.LOG_TAG, "Exception while reading text reply " + rte.toString());
531            }
532
533            String replyTag = isReply ? BODY_QUOTED_PART_REPLY : BODY_QUOTED_PART_FORWARD;
534            try {
535                addTextBodyPart(mp, "text/html", replyTag,
536                        EmailContent.Body.restoreReplyHtmlWithMessageId(context, localMessage.mId));
537            } catch (RuntimeException rte) {
538                Log.d(Email.LOG_TAG, "Exception while reading html reply " + rte.toString());
539            }
540
541            try {
542                addTextBodyPart(mp, "text/plain", replyTag,
543                        EmailContent.Body.restoreReplyTextWithMessageId(context, localMessage.mId));
544            } catch (RuntimeException rte) {
545                Log.d(Email.LOG_TAG, "Exception while reading text reply " + rte.toString());
546            }
547        }
548
549        // Attachments
550        // TODO: Make sure we deal with these as structures and don't accidentally upload files
551//        Uri uri = ContentUris.withAppendedId(Attachment.MESSAGE_ID_URI, localMessage.mId);
552//        Cursor attachments = context.getContentResolver().query(uri, Attachment.CONTENT_PROJECTION,
553//                null, null, null);
554//        try {
555//
556//        } finally {
557//            attachments.close();
558//        }
559
560        return message;
561    }
562
563    /**
564     * Helper method to add a body part for a given type of text, if found
565     *
566     * @param mp The text body part will be added to this multipart
567     * @param contentType The content-type of the text being added
568     * @param quotedPartTag If non-null, HEADER_ANDROID_BODY_QUOTED_PART will be set to this value
569     * @param partText The text to add.  If null, nothing happens
570     */
571    private static void addTextBodyPart(MimeMultipart mp, String contentType, String quotedPartTag,
572            String partText) throws MessagingException {
573        if (partText == null) {
574            return;
575        }
576        TextBody body = new TextBody(partText);
577        MimeBodyPart bp = new MimeBodyPart(body, contentType);
578        if (quotedPartTag != null) {
579            bp.addHeader(MimeHeader.HEADER_ANDROID_BODY_QUOTED_PART, quotedPartTag);
580        }
581        mp.addBodyPart(bp);
582    }
583
584    /**
585     * Conversion from provider account to legacy account
586     *
587     * Used for backup/restore.
588     *
589     * @param context application context
590     * @param fromAccount the provider account to be backed up (including transient hostauth's)
591     * @return a legacy Account object ready to be committed to preferences
592     */
593    /* package */ static Account makeLegacyAccount(Context context,
594            EmailContent.Account fromAccount) {
595        Account result = new Account(context);
596
597        result.setDescription(fromAccount.getDisplayName());
598        result.setEmail(fromAccount.getEmailAddress());
599        // fromAccount.mSyncKey - assume lost if restoring
600        result.setSyncWindow(fromAccount.getSyncLookback());
601        result.setAutomaticCheckIntervalMinutes(fromAccount.getSyncInterval());
602        // fromAccount.mHostAuthKeyRecv - id not saved; will be reassigned when restoring
603        // fromAccount.mHostAuthKeySend - id not saved; will be reassigned when restoring
604
605        // Provider Account flags, and how they are mapped.
606        //  FLAGS_NOTIFY_NEW_MAIL       -> mNotifyNewMail
607        //  FLAGS_VIBRATE               -> mVibrate
608        //  DELETE_POLICY_NEVER         -> mDeletePolicy
609        //  DELETE_POLICY_7DAYS
610        //  DELETE_POLICY_ON_DELETE
611        result.setNotifyNewMail(0 !=
612            (fromAccount.getFlags() & EmailContent.Account.FLAGS_NOTIFY_NEW_MAIL));
613        result.setVibrate(0 !=
614            (fromAccount.getFlags() & EmailContent.Account.FLAGS_VIBRATE));
615        result.setDeletePolicy(fromAccount.getDeletePolicy());
616
617        result.mUuid = fromAccount.getUuid();
618        result.setName(fromAccount.mSenderName);
619        result.setRingtone(fromAccount.mRingtoneUri);
620        result.mProtocolVersion = fromAccount.mProtocolVersion;
621        // int fromAccount.mNewMessageCount = will be reset on next sync
622        result.mSecurityFlags = fromAccount.mSecurityFlags;
623        result.mSignature = fromAccount.mSignature;
624
625        // Use the existing conversions from HostAuth <-> Uri
626        result.setStoreUri(fromAccount.getStoreUri(context));
627        result.setSenderUri(fromAccount.getSenderUri(context));
628
629        return result;
630    }
631
632    /**
633     * Conversion from legacy account to provider account
634     *
635     * Used for backup/restore and for account migration.
636     *
637     * @param context application context
638     * @param fromAccount the legacy account to convert to modern format
639     * @return an Account ready to be committed to provider
640     */
641    public static EmailContent.Account makeAccount(Context context, Account fromAccount) {
642
643        EmailContent.Account result = new EmailContent.Account();
644
645        result.setDisplayName(fromAccount.getDescription());
646        result.setEmailAddress(fromAccount.getEmail());
647        result.mSyncKey = null;
648        result.setSyncLookback(fromAccount.getSyncWindow());
649        result.setSyncInterval(fromAccount.getAutomaticCheckIntervalMinutes());
650        // result.mHostAuthKeyRecv;     -- will be set when object is saved
651        // result.mHostAuthKeySend;     -- will be set when object is saved
652        int flags = 0;
653        if (fromAccount.isNotifyNewMail())  flags |= EmailContent.Account.FLAGS_NOTIFY_NEW_MAIL;
654        if (fromAccount.isVibrate())        flags |= EmailContent.Account.FLAGS_VIBRATE;
655        result.setFlags(flags);
656        result.setDeletePolicy(fromAccount.getDeletePolicy());
657        // result.setDefaultAccount();  -- will be set by caller, if neededf
658        result.mCompatibilityUuid = fromAccount.getUuid();
659        result.setSenderName(fromAccount.getName());
660        result.setRingtone(fromAccount.getRingtone());
661        result.mProtocolVersion = fromAccount.mProtocolVersion;
662        result.mNewMessageCount = 0;
663        result.mSecurityFlags = fromAccount.mSecurityFlags;
664        result.mSecuritySyncKey = null;
665        result.mSignature = fromAccount.mSignature;
666
667        result.setStoreUri(context, fromAccount.getStoreUri());
668        result.setSenderUri(context, fromAccount.getSenderUri());
669
670        return result;
671    }
672
673    /**
674     * Conversion from legacy folder to provider mailbox.  Used for account migration.
675     * Note: Many mailbox fields are unused in IMAP & POP accounts.
676     *
677     * @param context application context
678     * @param toAccount the provider account that this folder will be associated with
679     * @param fromFolder the legacy folder to convert to modern format
680     * @return an Account ready to be committed to provider
681     */
682    public static EmailContent.Mailbox makeMailbox(Context context, EmailContent.Account toAccount,
683            Folder fromFolder) throws MessagingException {
684        EmailContent.Mailbox result = new EmailContent.Mailbox();
685
686        result.mDisplayName = fromFolder.getName();
687        // result.mServerId
688        // result.mParentServerId
689        result.mAccountKey = toAccount.mId;
690        result.mType = inferMailboxTypeFromName(context, fromFolder.getName());
691        // result.mDelimiter
692        // result.mSyncKey
693        // result.mSyncLookback
694        // result.mSyncInterval
695        result.mSyncTime = 0;
696        result.mUnreadCount = fromFolder.getUnreadMessageCount();
697        result.mFlagVisible = true;
698        result.mFlags = 0;
699        result.mVisibleLimit = Email.VISIBLE_LIMIT_DEFAULT;
700        // result.mSyncStatus
701
702        return result;
703    }
704
705    /**
706     * Infer mailbox type from mailbox name.  Used by MessagingController (for live folder sync)
707     * and for legacy account upgrades.
708     */
709    public static synchronized int inferMailboxTypeFromName(Context context, String mailboxName) {
710        if (sServerMailboxNames.size() == 0) {
711            // preload the hashmap, one time only
712            sServerMailboxNames.put(
713                    context.getString(R.string.mailbox_name_server_inbox).toLowerCase(),
714                    Mailbox.TYPE_INBOX);
715            sServerMailboxNames.put(
716                    context.getString(R.string.mailbox_name_server_outbox).toLowerCase(),
717                    Mailbox.TYPE_OUTBOX);
718            sServerMailboxNames.put(
719                    context.getString(R.string.mailbox_name_server_drafts).toLowerCase(),
720                    Mailbox.TYPE_DRAFTS);
721            sServerMailboxNames.put(
722                    context.getString(R.string.mailbox_name_server_trash).toLowerCase(),
723                    Mailbox.TYPE_TRASH);
724            sServerMailboxNames.put(
725                    context.getString(R.string.mailbox_name_server_sent).toLowerCase(),
726                    Mailbox.TYPE_SENT);
727            sServerMailboxNames.put(
728                    context.getString(R.string.mailbox_name_server_junk).toLowerCase(),
729                    Mailbox.TYPE_JUNK);
730        }
731        if (mailboxName == null || mailboxName.length() == 0) {
732            return EmailContent.Mailbox.TYPE_MAIL;
733        }
734        String lowerCaseName = mailboxName.toLowerCase();
735        Integer type = sServerMailboxNames.get(lowerCaseName);
736        if (type != null) {
737            return type;
738        }
739        return EmailContent.Mailbox.TYPE_MAIL;
740    }
741}
742