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