Home | History | Annotate | Download | only in dictionarypack
      1 /**
      2  * Copyright (C) 2011 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License"); you may not
      5  * use this file except in compliance with the License. You may obtain a copy of
      6  * 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, WITHOUT
     12  * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
     13  * License for the specific language governing permissions and limitations under
     14  * the License.
     15  */
     16 
     17 package com.android.inputmethod.dictionarypack;
     18 
     19 import android.app.AlarmManager;
     20 import android.app.PendingIntent;
     21 import android.app.Service;
     22 import android.content.Context;
     23 import android.content.Intent;
     24 import android.os.IBinder;
     25 import android.text.format.DateUtils;
     26 import android.util.Log;
     27 import android.widget.Toast;
     28 
     29 import com.android.inputmethod.latin.R;
     30 
     31 import java.util.Locale;
     32 import java.util.Random;
     33 
     34 /**
     35  * Service that handles background tasks for the dictionary provider.
     36  *
     37  * This service provides the context for the long-running operations done by the
     38  * dictionary provider. Those include:
     39  * - Checking for the last update date and scheduling the next update. This runs every
     40  *   day around midnight, upon reception of the DATE_CHANGED_INTENT_ACTION broadcast.
     41  *   Every four days, it schedules an update of the metadata with the alarm manager.
     42  * - Issuing the order to update the metadata. This runs every four days, between 0 and
     43  *   6, upon reception of the UPDATE_NOW_INTENT_ACTION broadcast sent by the alarm manager
     44  *   as a result of the above action.
     45  * - Handling a download that just ended. These come in two flavors:
     46  *   - Metadata is finished downloading. We should check whether there are new dictionaries
     47  *     available, and download those that we need that have new versions.
     48  *   - A dictionary file finished downloading. We should put the file ready for a client IME
     49  *     to access, and mark the current state as such.
     50  */
     51 public final class DictionaryService extends Service {
     52     private static final String TAG = DictionaryService.class.getName();
     53 
     54     /**
     55      * The package name, to use in the intent actions.
     56      */
     57     private static final String PACKAGE_NAME = "com.android.android.inputmethod.latin";
     58 
     59     /**
     60      * The action of the intent to tell the dictionary provider to update now.
     61      */
     62     private static final String UPDATE_NOW_INTENT_ACTION = PACKAGE_NAME + ".UPDATE_NOW";
     63 
     64     /**
     65      * The action of the date changing, used to schedule a periodic freshness check
     66      */
     67     private static final String DATE_CHANGED_INTENT_ACTION =
     68             Intent.ACTION_DATE_CHANGED;
     69 
     70     /**
     71      * The action of displaying a toast to warn the user an automatic download is starting.
     72      */
     73     /* package */ static final String SHOW_DOWNLOAD_TOAST_INTENT_ACTION =
     74             PACKAGE_NAME + ".SHOW_DOWNLOAD_TOAST_INTENT_ACTION";
     75 
     76     /**
     77      * A locale argument, as a String.
     78      */
     79     /* package */ static final String LOCALE_INTENT_ARGUMENT = "locale";
     80 
     81     /**
     82      * How often, in milliseconds, we want to update the metadata. This is a
     83      * floor value; actually, it may happen several hours later, or even more.
     84      */
     85     private static final long UPDATE_FREQUENCY = 4 * DateUtils.DAY_IN_MILLIS;
     86 
     87     /**
     88      * We are waked around midnight, local time. We want to wake between midnight and 6 am,
     89      * roughly. So use a random time between 0 and this delay.
     90      */
     91     private static final int MAX_ALARM_DELAY = 6 * ((int)AlarmManager.INTERVAL_HOUR);
     92 
     93     /**
     94      * How long we consider a "very long time". If no update took place in this time,
     95      * the content provider will trigger an update in the background.
     96      */
     97     private static final long VERY_LONG_TIME = 14 * DateUtils.DAY_IN_MILLIS;
     98 
     99     /**
    100      * The last seen start Id. This must be stored because we must only call stopSelfResult() with
    101      * the last seen Id, or the service won't stop.
    102      */
    103     private int mLastSeenStartId;
    104 
    105     /**
    106      * The command count. We need this because we need to not call stopSelfResult() while we still
    107      * have commands running.
    108      */
    109     private int mCommandCount;
    110 
    111     @Override
    112     public void onCreate() {
    113         mLastSeenStartId = 0;
    114         mCommandCount = 0;
    115     }
    116 
    117     @Override
    118     public void onDestroy() {
    119     }
    120 
    121     @Override
    122     public IBinder onBind(Intent intent) {
    123         // This service cannot be bound
    124         return null;
    125     }
    126 
    127     /**
    128      * Executes an explicit command.
    129      *
    130      * This is the entry point for arbitrary commands that are executed upon reception of certain
    131      * events that should be executed on the context of this service. The supported commands are:
    132      * - Check last update time and possibly schedule an update of the data for later.
    133      *     This is triggered every day, upon reception of the DATE_CHANGED_INTENT_ACTION broadcast.
    134      * - Update data NOW.
    135      *     This is normally received upon trigger of the scheduled update.
    136      * - Handle a finished download.
    137      *     This executes the actions that must be taken after a file (metadata or dictionary data
    138      *     has been downloaded (or failed to download).
    139      */
    140     @Override
    141     public synchronized int onStartCommand(final Intent intent, final int flags,
    142             final int startId) {
    143         final DictionaryService self = this;
    144         mLastSeenStartId = startId;
    145         mCommandCount += 1;
    146         if (SHOW_DOWNLOAD_TOAST_INTENT_ACTION.equals(intent.getAction())) {
    147             // This is a UI action, it can't be run in another thread
    148             showStartDownloadingToast(this, LocaleUtils.constructLocaleFromString(
    149                     intent.getStringExtra(LOCALE_INTENT_ARGUMENT)));
    150         } else {
    151             // If it's a command that does not require UI, create a thread to do the work
    152             // and return right away. DATE_CHANGED or UPDATE_NOW are examples of such commands.
    153             new Thread("updateOrFinishDownload") {
    154                 @Override
    155                 public void run() {
    156                     dispatchBroadcast(self, intent);
    157                     synchronized(self) {
    158                         if (--mCommandCount <= 0) {
    159                             if (!stopSelfResult(mLastSeenStartId)) {
    160                                 Log.e(TAG, "Can't stop ourselves");
    161                             }
    162                         }
    163                     }
    164                 }
    165             }.start();
    166         }
    167         return Service.START_REDELIVER_INTENT;
    168     }
    169 
    170     private static void dispatchBroadcast(final Context context, final Intent intent) {
    171         if (DATE_CHANGED_INTENT_ACTION.equals(intent.getAction())) {
    172             // This happens when the date of the device changes. This normally happens
    173             // at midnight local time, but it may happen if the user changes the date
    174             // by hand or something similar happens.
    175             checkTimeAndMaybeSetupUpdateAlarm(context);
    176         } else if (UPDATE_NOW_INTENT_ACTION.equals(intent.getAction())) {
    177             // Intent to trigger an update now.
    178             UpdateHandler.update(context, false);
    179         } else {
    180             UpdateHandler.downloadFinished(context, intent);
    181         }
    182     }
    183 
    184     /**
    185      * Setups an alarm to check for updates if an update is due.
    186      */
    187     private static void checkTimeAndMaybeSetupUpdateAlarm(final Context context) {
    188         // Of all clients, if the one that hasn't been updated for the longest
    189         // is still more recent than UPDATE_FREQUENCY, do nothing.
    190         if (!isLastUpdateAtLeastThisOld(context, UPDATE_FREQUENCY)) return;
    191 
    192         PrivateLog.log("Date changed - registering alarm");
    193         AlarmManager alarmManager = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
    194 
    195         // Best effort to wake between midnight and MAX_ALARM_DELAY in the morning.
    196         // It doesn't matter too much if this is very inexact.
    197         final long now = System.currentTimeMillis();
    198         final long alarmTime = now + new Random().nextInt(MAX_ALARM_DELAY);
    199         final Intent updateIntent = new Intent(DictionaryService.UPDATE_NOW_INTENT_ACTION);
    200         final PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0,
    201                 updateIntent, PendingIntent.FLAG_CANCEL_CURRENT);
    202 
    203         // We set the alarm in the type that doesn't forcefully wake the device
    204         // from sleep, but fires the next time the device actually wakes for any
    205         // other reason.
    206         if (null != alarmManager) alarmManager.set(AlarmManager.RTC, alarmTime, pendingIntent);
    207     }
    208 
    209     /**
    210      * Utility method to decide whether the last update is older than a certain time.
    211      *
    212      * @return true if at least `time' milliseconds have elapsed since last update, false otherwise.
    213      */
    214     private static boolean isLastUpdateAtLeastThisOld(final Context context, final long time) {
    215         final long now = System.currentTimeMillis();
    216         final long lastUpdate = MetadataDbHelper.getOldestUpdateTime(context);
    217         PrivateLog.log("Last update was " + lastUpdate);
    218         return lastUpdate + time < now;
    219     }
    220 
    221     /**
    222      * Refreshes data if it hasn't been refreshed in a very long time.
    223      *
    224      * This will check the last update time, and if it's been more than VERY_LONG_TIME,
    225      * update metadata now - and possibly take subsequent update actions.
    226      */
    227     public static void updateNowIfNotUpdatedInAVeryLongTime(final Context context) {
    228         if (!isLastUpdateAtLeastThisOld(context, VERY_LONG_TIME)) return;
    229         UpdateHandler.update(context, false);
    230     }
    231 
    232     /**
    233      * Shows a toast informing the user that an automatic dictionary download is starting.
    234      */
    235     private static void showStartDownloadingToast(final Context context, final Locale locale) {
    236         final String toastText = String.format(
    237                 context.getString(R.string.toast_downloading_suggestions),
    238                 locale.getDisplayName());
    239         Toast.makeText(context, toastText, Toast.LENGTH_LONG).show();
    240     }
    241 }
    242