1d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd/* 2d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * Copyright (C) 2015 The Android Open Source Project 3d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * 4d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * Licensed under the Apache License, Version 2.0 (the "License"); 5d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * you may not use this file except in compliance with the License. 6d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * You may obtain a copy of the License at 7d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * 8d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * http://www.apache.org/licenses/LICENSE-2.0 9d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * 10d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * Unless required by applicable law or agreed to in writing, software 11d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * distributed under the License is distributed on an "AS IS" BASIS, 12d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * See the License for the specific language governing permissions and 14d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * limitations under the License. 15d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd */ 16d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 17d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddpackage com.android.messaging.datamodel.action; 18d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 19d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddimport android.content.ContentValues; 20d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddimport android.os.Parcel; 21d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddimport android.os.Parcelable; 22d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 23d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddimport com.android.messaging.datamodel.DataModel; 24d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddimport com.android.messaging.datamodel.DatabaseHelper; 25d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddimport com.android.messaging.datamodel.DatabaseWrapper; 26d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddimport com.android.messaging.datamodel.data.MessageData; 27d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddimport com.android.messaging.util.LogUtil; 28d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 29d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd/** 30d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * Action used to fixup actively downloading or sending status at startup - just in case we 31d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd * crash - never run this when a message might actually be sending or downloading. 32d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd */ 33d3b009ae55651f1e60950342468e3c37fdeb0796Mike Doddpublic class FixupMessageStatusOnStartupAction extends Action implements Parcelable { 34d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd private static final String TAG = LogUtil.BUGLE_DATAMODEL_TAG; 35d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 36d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd public static void fixupMessageStatus() { 37d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd final FixupMessageStatusOnStartupAction action = new FixupMessageStatusOnStartupAction(); 38d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd action.start(); 39d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } 40d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 41d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd private FixupMessageStatusOnStartupAction() { 42d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } 43d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 44d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd @Override 45d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd protected Object executeAction() { 46d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // Now mark any messages in active sending or downloading state as inactive 47d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd final DatabaseWrapper db = DataModel.get().getDatabase(); 48d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd db.beginTransaction(); 49d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd int downloadFailedCnt = 0; 50d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd int sendFailedCnt = 0; 51d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd try { 52d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // For both sending and downloading messages, let's assume they failed. 53d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // For MMS sent/downloaded via platform, the sent/downloaded pending intent 54d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // may come back. That will update the message. User may see the message 55d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // in wrong status within a short window if that happens. But this should 56d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // rarely happen. This is a simple solution to situations like app gets killed 57d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // while the pending intent is still in the fly. Alternatively, we could 58d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // keep the status for platform sent/downloaded MMS and timeout these messages. 59d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // But that is much more complex. 60d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd final ContentValues values = new ContentValues(); 61d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd values.put(DatabaseHelper.MessageColumns.STATUS, 62d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd MessageData.BUGLE_STATUS_INCOMING_DOWNLOAD_FAILED); 63d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd downloadFailedCnt += db.update(DatabaseHelper.MESSAGES_TABLE, values, 64d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd DatabaseHelper.MessageColumns.STATUS + " IN (?, ?)", 65d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd new String[]{ 66d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd Integer.toString(MessageData.BUGLE_STATUS_INCOMING_AUTO_DOWNLOADING), 67d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd Integer.toString(MessageData.BUGLE_STATUS_INCOMING_MANUAL_DOWNLOADING) 68d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd }); 69d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd values.clear(); 70d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 71d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd values.clear(); 72d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd values.put(DatabaseHelper.MessageColumns.STATUS, 73d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd MessageData.BUGLE_STATUS_OUTGOING_FAILED); 74d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd sendFailedCnt = db.update(DatabaseHelper.MESSAGES_TABLE, values, 75d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd DatabaseHelper.MessageColumns.STATUS + " IN (?, ?)", 76d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd new String[]{ 77d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd Integer.toString(MessageData.BUGLE_STATUS_OUTGOING_SENDING), 78d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd Integer.toString(MessageData.BUGLE_STATUS_OUTGOING_RESENDING) 79d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd }); 80d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 81d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd db.setTransactionSuccessful(); 82d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } finally { 83d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd db.endTransaction(); 84d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } 85d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 86d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd LogUtil.i(TAG, "Fixup: Send failed - " + sendFailedCnt 87d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd + " Download failed - " + downloadFailedCnt); 88d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 89d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd // Don't send contentObserver notifications as displayed text should not change 90d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd return null; 91d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } 92d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 93d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd private FixupMessageStatusOnStartupAction(final Parcel in) { 94d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd super(in); 95d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } 96d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 97d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd public static final Parcelable.Creator<FixupMessageStatusOnStartupAction> CREATOR 98d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd = new Parcelable.Creator<FixupMessageStatusOnStartupAction>() { 99d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd @Override 100d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd public FixupMessageStatusOnStartupAction createFromParcel(final Parcel in) { 101d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd return new FixupMessageStatusOnStartupAction(in); 102d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } 103d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 104d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd @Override 105d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd public FixupMessageStatusOnStartupAction[] newArray(final int size) { 106d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd return new FixupMessageStatusOnStartupAction[size]; 107d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } 108d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd }; 109d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd 110d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd @Override 111d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd public void writeToParcel(final Parcel parcel, final int flags) { 112d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd writeActionToParcel(parcel, flags); 113d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd } 114d3b009ae55651f1e60950342468e3c37fdeb0796Mike Dodd} 115