Home | History | Annotate | Download | only in setup
      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 
     17 package com.android.email.activity.setup;
     18 
     19 import android.content.ContentResolver;
     20 import android.content.ContentValues;
     21 import android.content.Context;
     22 import android.content.res.XmlResourceParser;
     23 import android.text.Editable;
     24 import android.text.TextUtils;
     25 import android.widget.EditText;
     26 
     27 import com.android.email.R;
     28 import com.android.email.provider.AccountBackupRestore;
     29 import com.android.emailcommon.Logging;
     30 import com.android.emailcommon.VendorPolicyLoader;
     31 import com.android.emailcommon.VendorPolicyLoader.Provider;
     32 import com.android.emailcommon.provider.Account;
     33 import com.android.emailcommon.provider.EmailContent.AccountColumns;
     34 import com.android.emailcommon.provider.QuickResponse;
     35 import com.android.emailcommon.utility.Utility;
     36 import com.android.mail.utils.LogUtils;
     37 import com.google.common.annotations.VisibleForTesting;
     38 
     39 public class AccountSettingsUtils {
     40 
     41     /** Pattern to match any part of a domain */
     42     private final static String WILD_STRING = "*";
     43     /** Will match any, single character */
     44     private final static char WILD_CHARACTER = '?';
     45     private final static String DOMAIN_SEPARATOR = "\\.";
     46 
     47     /**
     48      * Commits the UI-related settings of an account to the provider.  This is static so that it
     49      * can be used by the various account activities.  If the account has never been saved, this
     50      * method saves it; otherwise, it just saves the settings.
     51      * @param context the context of the caller
     52      * @param account the account whose settings will be committed
     53      */
     54     public static void commitSettings(Context context, Account account) {
     55         if (!account.isSaved()) {
     56             account.save(context);
     57 
     58             // Set up default quick responses here...
     59             String[] defaultQuickResponses =
     60                 context.getResources().getStringArray(R.array.default_quick_responses);
     61             ContentValues cv = new ContentValues();
     62             cv.put(QuickResponse.ACCOUNT_KEY, account.mId);
     63             ContentResolver resolver = context.getContentResolver();
     64             for (String quickResponse: defaultQuickResponses) {
     65                 // Allow empty entries (some localizations may not want to have the maximum
     66                 // number)
     67                 if (!TextUtils.isEmpty(quickResponse)) {
     68                     cv.put(QuickResponse.TEXT, quickResponse);
     69                     resolver.insert(QuickResponse.CONTENT_URI, cv);
     70                 }
     71             }
     72         } else {
     73             ContentValues cv = getAccountContentValues(account);
     74             account.update(context, cv);
     75         }
     76 
     77         // Update the backup (side copy) of the accounts
     78         AccountBackupRestore.backup(context);
     79     }
     80 
     81     /**
     82      * Returns a set of content values to commit account changes (not including the foreign keys
     83      * for the two host auth's and policy) to the database.  Does not actually commit anything.
     84      */
     85     public static ContentValues getAccountContentValues(Account account) {
     86         ContentValues cv = new ContentValues();
     87         cv.put(AccountColumns.DISPLAY_NAME, account.getDisplayName());
     88         cv.put(AccountColumns.SENDER_NAME, account.getSenderName());
     89         cv.put(AccountColumns.SIGNATURE, account.getSignature());
     90         cv.put(AccountColumns.SYNC_INTERVAL, account.mSyncInterval);
     91         cv.put(AccountColumns.FLAGS, account.mFlags);
     92         cv.put(AccountColumns.SYNC_LOOKBACK, account.mSyncLookback);
     93         cv.put(AccountColumns.SECURITY_SYNC_KEY, account.mSecuritySyncKey);
     94         return cv;
     95     }
     96 
     97     /**
     98      * Search the list of known Email providers looking for one that matches the user's email
     99      * domain.  We check for vendor supplied values first, then we look in providers_product.xml,
    100      * and finally by the entries in platform providers.xml.  This provides a nominal override
    101      * capability.
    102      *
    103      * A match is defined as any provider entry for which the "domain" attribute matches.
    104      *
    105      * @param domain The domain portion of the user's email address
    106      * @return suitable Provider definition, or null if no match found
    107      */
    108     public static Provider findProviderForDomain(Context context, String domain) {
    109         Provider p = VendorPolicyLoader.getInstance(context).findProviderForDomain(domain);
    110         if (p == null) {
    111             p = findProviderForDomain(context, domain, R.xml.providers_product);
    112         }
    113         if (p == null) {
    114             p = findProviderForDomain(context, domain, R.xml.providers);
    115         }
    116         return p;
    117     }
    118 
    119     /**
    120      * Search a single resource containing known Email provider definitions.
    121      *
    122      * @param domain The domain portion of the user's email address
    123      * @param resourceId Id of the provider resource to scan
    124      * @return suitable Provider definition, or null if no match found
    125      */
    126     /*package*/ static Provider findProviderForDomain(
    127             Context context, String domain, int resourceId) {
    128         try {
    129             XmlResourceParser xml = context.getResources().getXml(resourceId);
    130             int xmlEventType;
    131             Provider provider = null;
    132             while ((xmlEventType = xml.next()) != XmlResourceParser.END_DOCUMENT) {
    133                 if (xmlEventType == XmlResourceParser.START_TAG
    134                         && "provider".equals(xml.getName())) {
    135                     String providerDomain = getXmlAttribute(context, xml, "domain");
    136                     try {
    137                         if (matchProvider(domain, providerDomain)) {
    138                             provider = new Provider();
    139                             provider.id = getXmlAttribute(context, xml, "id");
    140                             provider.label = getXmlAttribute(context, xml, "label");
    141                             provider.domain = domain.toLowerCase();
    142                             provider.note = getXmlAttribute(context, xml, "note");
    143                         }
    144                     } catch (IllegalArgumentException e) {
    145                         LogUtils.w(Logging.LOG_TAG, "providers line: " + xml.getLineNumber() +
    146                                 "; Domain contains multiple globals");
    147                     }
    148                 }
    149                 else if (xmlEventType == XmlResourceParser.START_TAG
    150                         && "incoming".equals(xml.getName())
    151                         && provider != null) {
    152                     provider.incomingUriTemplate = getXmlAttribute(context, xml, "uri");
    153                     provider.incomingUsernameTemplate = getXmlAttribute(context, xml, "username");
    154                 }
    155                 else if (xmlEventType == XmlResourceParser.START_TAG
    156                         && "outgoing".equals(xml.getName())
    157                         && provider != null) {
    158                     provider.outgoingUriTemplate = getXmlAttribute(context, xml, "uri");
    159                     provider.outgoingUsernameTemplate = getXmlAttribute(context, xml, "username");
    160                 }
    161                 else if (xmlEventType == XmlResourceParser.END_TAG
    162                         && "provider".equals(xml.getName())
    163                         && provider != null) {
    164                     return provider;
    165                 }
    166             }
    167         }
    168         catch (Exception e) {
    169             LogUtils.e(Logging.LOG_TAG, "Error while trying to load provider settings.", e);
    170         }
    171         return null;
    172     }
    173 
    174     /**
    175      * Returns true if the string <code>s1</code> matches the string <code>s2</code>. The string
    176      * <code>s2</code> may contain any number of wildcards -- a '?' character -- and/or asterisk
    177      * characters -- '*'. Wildcards match any single character, while the asterisk matches a domain
    178      * part (i.e. substring demarcated by a period, '.')
    179      */
    180     @VisibleForTesting
    181     public static boolean matchProvider(String testDomain, String providerDomain) {
    182         String[] testParts = testDomain.split(DOMAIN_SEPARATOR);
    183         String[] providerParts = providerDomain.split(DOMAIN_SEPARATOR);
    184         if (testParts.length != providerParts.length) {
    185             return false;
    186         }
    187         for (int i = 0; i < testParts.length; i++) {
    188             String testPart = testParts[i].toLowerCase();
    189             String providerPart = providerParts[i].toLowerCase();
    190             if (!providerPart.equals(WILD_STRING) &&
    191                     !matchWithWildcards(testPart, providerPart)) {
    192                 return false;
    193             }
    194         }
    195         return true;
    196     }
    197 
    198     private static boolean matchWithWildcards(String testPart, String providerPart) {
    199         int providerLength = providerPart.length();
    200         if (testPart.length() != providerLength){
    201             return false;
    202         }
    203         for (int i = 0; i < providerLength; i++) {
    204             char testChar = testPart.charAt(i);
    205             char providerChar = providerPart.charAt(i);
    206             if (testChar != providerChar && providerChar != WILD_CHARACTER) {
    207                 return false;
    208             }
    209         }
    210         return true;
    211     }
    212 
    213     /**
    214      * Attempts to get the given attribute as a String resource first, and if it fails
    215      * returns the attribute as a simple String value.
    216      * @param xml
    217      * @param name
    218      * @return the requested resource
    219      */
    220     private static String getXmlAttribute(Context context, XmlResourceParser xml, String name) {
    221         int resId = xml.getAttributeResourceValue(null, name, 0);
    222         if (resId == 0) {
    223             return xml.getAttributeValue(null, name);
    224         }
    225         else {
    226             return context.getString(resId);
    227         }
    228     }
    229 
    230     /**
    231      * Infer potential email server addresses from domain names
    232      *
    233      * Incoming: Prepend "imap" or "pop3" to domain, unless "pop", "pop3",
    234      *          "imap", or "mail" are found.
    235      * Outgoing: Prepend "smtp" if domain starts with any in the host prefix array
    236      *
    237      * @param server name as we know it so far
    238      * @param incoming "pop3" or "imap" (or null)
    239      * @param outgoing "smtp" or null
    240      * @return the post-processed name for use in the UI
    241      */
    242     public static String inferServerName(Context context, String server, String incoming,
    243             String outgoing) {
    244         // Default values cause entire string to be kept, with prepended server string
    245         int keepFirstChar = 0;
    246         int firstDotIndex = server.indexOf('.');
    247         if (firstDotIndex != -1) {
    248             // look at first word and decide what to do
    249             String firstWord = server.substring(0, firstDotIndex).toLowerCase();
    250             String[] hostPrefixes =
    251                     context.getResources().getStringArray(R.array.smtp_host_prefixes);
    252             boolean canSubstituteSmtp = Utility.arrayContains(hostPrefixes, firstWord);
    253             boolean isMail = "mail".equals(firstWord);
    254             // Now decide what to do
    255             if (incoming != null) {
    256                 // For incoming, we leave imap/pop/pop3/mail alone, or prepend incoming
    257                 if (canSubstituteSmtp || isMail) {
    258                     return server;
    259                 }
    260             } else {
    261                 // For outgoing, replace imap/pop/pop3 with outgoing, leave mail alone, or
    262                 // prepend outgoing
    263                 if (canSubstituteSmtp) {
    264                     keepFirstChar = firstDotIndex + 1;
    265                 } else if (isMail) {
    266                     return server;
    267                 } else {
    268                     // prepend
    269                 }
    270             }
    271         }
    272         return ((incoming != null) ? incoming : outgoing) + '.' + server.substring(keepFirstChar);
    273     }
    274 
    275     /**
    276      * Helper to set error status on password fields that have leading or trailing spaces
    277      */
    278     public static void checkPasswordSpaces(Context context, EditText passwordField) {
    279         Editable password = passwordField.getText();
    280         int length = password.length();
    281         if (length > 0) {
    282             if (password.charAt(0) == ' ' || password.charAt(length-1) == ' ') {
    283                 passwordField.setError(context.getString(R.string.account_password_spaces_error));
    284             }
    285         }
    286     }
    287 
    288 }
    289