/*
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.contacts;
import static android.app.PendingIntent.FLAG_IMMUTABLE;
import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.content.OperationApplicationException;
import android.os.AsyncTask;
import android.os.IBinder;
import android.os.RemoteException;
import androidx.annotation.Nullable;
import androidx.core.app.NotificationCompat;
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
import android.util.TimingLogger;
import com.android.contacts.activities.PeopleActivity;
import com.android.contacts.database.SimContactDao;
import com.android.contacts.model.SimCard;
import com.android.contacts.model.SimContact;
import com.android.contacts.model.account.AccountWithDataSet;
import com.android.contacts.util.ContactsNotificationChannelsUtil;
import com.android.contactsbind.FeedbackHelper;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/**
* Imports {@link SimContact}s from a background thread
*/
public class SimImportService extends Service {
private static final String TAG = "SimImportService";
/**
* Wrapper around the service state for testability
*/
public interface StatusProvider {
/**
* Returns whether there is any imports still pending
*
*
This should be called from the UI thread
*/
boolean isRunning();
/**
* Returns whether an import for sim has been requested
*
* This should be called from the UI thread
*/
boolean isImporting(SimCard sim);
}
public static final String EXTRA_ACCOUNT = "account";
public static final String EXTRA_SIM_CONTACTS = "simContacts";
public static final String EXTRA_SIM_SUBSCRIPTION_ID = "simSubscriptionId";
public static final String EXTRA_RESULT_CODE = "resultCode";
public static final String EXTRA_RESULT_COUNT = "count";
public static final String EXTRA_OPERATION_REQUESTED_AT_TIME = "requestedTime";
public static final String BROADCAST_SERVICE_STATE_CHANGED =
SimImportService.class.getName() + "#serviceStateChanged";
public static final String BROADCAST_SIM_IMPORT_COMPLETE =
SimImportService.class.getName() + "#simImportComplete";
public static final int RESULT_UNKNOWN = 0;
public static final int RESULT_SUCCESS = 1;
public static final int RESULT_FAILURE = 2;
// VCardService uses jobIds for it's notifications which count up from 0 so we just use a
// bigger number to prevent overlap.
private static final int NOTIFICATION_ID = 100;
private ExecutorService mExecutor = Executors.newSingleThreadExecutor();
// Keeps track of current tasks. This is only modified from the UI thread.
private static List sPending = new ArrayList<>();
private static StatusProvider sStatusProvider = new StatusProvider() {
@Override
public boolean isRunning() {
return !sPending.isEmpty();
}
@Override
public boolean isImporting(SimCard sim) {
return SimImportService.isImporting(sim);
}
};
/**
* Returns whether an import for sim has been requested
*
* This should be called from the UI thread
*/
private static boolean isImporting(SimCard sim) {
for (ImportTask task : sPending) {
if (task.getSim().equals(sim)) {
return true;
}
}
return false;
}
public static StatusProvider getStatusProvider() {
return sStatusProvider;
}
/**
* Starts an import of the contacts from the sim into the target account
*
* @param context context to use for starting the service
* @param subscriptionId the subscriptionId of the SIM card that is being imported. See
* {@link android.telephony.SubscriptionInfo#getSubscriptionId()}.
* Upon completion the SIM for that subscription ID will be marked as
* imported
* @param contacts the contacts to import
* @param targetAccount the account import the contacts into
*/
public static void startImport(Context context, int subscriptionId,
ArrayList contacts, AccountWithDataSet targetAccount) {
context.startService(new Intent(context, SimImportService.class)
.putExtra(EXTRA_SIM_CONTACTS, contacts)
.putExtra(EXTRA_SIM_SUBSCRIPTION_ID, subscriptionId)
.putExtra(EXTRA_ACCOUNT, targetAccount));
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public int onStartCommand(Intent intent, int flags, final int startId) {
ContactsNotificationChannelsUtil.createDefaultChannel(this);
final ImportTask task = createTaskForIntent(intent, startId);
if (task == null) {
new StopTask(this, startId).executeOnExecutor(mExecutor);
return START_NOT_STICKY;
}
sPending.add(task);
task.executeOnExecutor(mExecutor);
notifyStateChanged();
return START_REDELIVER_INTENT;
}
@Override
public void onDestroy() {
super.onDestroy();
mExecutor.shutdown();
}
private ImportTask createTaskForIntent(Intent intent, int startId) {
final AccountWithDataSet targetAccount = intent.getParcelableExtra(EXTRA_ACCOUNT);
final ArrayList contacts =
intent.getParcelableArrayListExtra(EXTRA_SIM_CONTACTS);
final int subscriptionId = intent.getIntExtra(EXTRA_SIM_SUBSCRIPTION_ID,
SimCard.NO_SUBSCRIPTION_ID);
final SimContactDao dao = SimContactDao.create(this);
final SimCard sim = dao.getSimBySubscriptionId(subscriptionId);
if (sim != null) {
return new ImportTask(sim, contacts, targetAccount, dao, startId);
} else {
return null;
}
}
private Notification getCompletedNotification() {
final Intent intent = new Intent(this, PeopleActivity.class);
final NotificationCompat.Builder builder = new NotificationCompat.Builder(
this, ContactsNotificationChannelsUtil.DEFAULT_CHANNEL);
builder.setOngoing(false)
.setAutoCancel(true)
.setContentTitle(this.getString(R.string.importing_sim_finished_title))
.setColor(this.getResources().getColor(R.color.dialtacts_theme_color))
.setSmallIcon(R.drawable.quantum_ic_done_vd_theme_24)
.setContentIntent(PendingIntent.getActivity(this, 0, intent, FLAG_IMMUTABLE));
return builder.build();
}
private Notification getFailedNotification() {
final Intent intent = new Intent(this, PeopleActivity.class);
final NotificationCompat.Builder builder = new NotificationCompat.Builder(
this, ContactsNotificationChannelsUtil.DEFAULT_CHANNEL);
builder.setOngoing(false)
.setAutoCancel(true)
.setContentTitle(this.getString(R.string.importing_sim_failed_title))
.setContentText(this.getString(R.string.importing_sim_failed_message))
.setColor(this.getResources().getColor(R.color.dialtacts_theme_color))
.setSmallIcon(R.drawable.quantum_ic_error_vd_theme_24)
.setContentIntent(PendingIntent.getActivity(this, 0, intent, FLAG_IMMUTABLE));
return builder.build();
}
private Notification getImportingNotification() {
final NotificationCompat.Builder builder = new NotificationCompat.Builder(
this, ContactsNotificationChannelsUtil.DEFAULT_CHANNEL);
final String description = getString(R.string.importing_sim_in_progress_title);
builder.setOngoing(true)
.setProgress(/* current */ 0, /* max */ 100, /* indeterminate */ true)
.setContentTitle(description)
.setColor(this.getResources().getColor(R.color.dialtacts_theme_color))
.setSmallIcon(android.R.drawable.stat_sys_download);
return builder.build();
}
private void notifyStateChanged() {
LocalBroadcastManager.getInstance(this).sendBroadcast(
new Intent(BROADCAST_SERVICE_STATE_CHANGED));
}
// Schedule a task that calls stopSelf when it completes. This is used to ensure that the
// calls to stopSelf occur in the correct order (because this service uses a single thread
// executor this won't run until all work that was requested before it has finished)
private static class StopTask extends AsyncTask {
private Service mHost;
private final int mStartId;
private StopTask(Service host, int startId) {
mHost = host;
mStartId = startId;
}
@Override
protected Void doInBackground(Void... params) {
return null;
}
@Override
protected void onPostExecute(Void aVoid) {
super.onPostExecute(aVoid);
mHost.stopSelf(mStartId);
}
}
private class ImportTask extends AsyncTask {
private final SimCard mSim;
private final List mContacts;
private final AccountWithDataSet mTargetAccount;
private final SimContactDao mDao;
private final NotificationManager mNotificationManager;
private final int mStartId;
private final long mStartTime;
public ImportTask(SimCard sim, List contacts, AccountWithDataSet targetAccount,
SimContactDao dao, int startId) {
mSim = sim;
mContacts = contacts;
mTargetAccount = targetAccount;
mDao = dao;
mNotificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
mStartId = startId;
mStartTime = System.currentTimeMillis();
}
@Override
protected void onPreExecute() {
super.onPreExecute();
startForeground(NOTIFICATION_ID, getImportingNotification());
}
@Override
protected Boolean doInBackground(Void... params) {
final TimingLogger timer = new TimingLogger(TAG, "import");
try {
// Just import them all at once.
// Experimented with using smaller batches (e.g. 25 and 50) so that percentage
// progress could be displayed however this slowed down the import by over a factor
// of 2. If the batch size is over a 100 then most cases will only require a single
// batch so we don't even worry about displaying accurate progress
mDao.importContacts(mContacts, mTargetAccount);
mDao.persistSimState(mSim.withImportedState(true));
timer.addSplit("done");
timer.dumpToLog();
} catch (RemoteException|OperationApplicationException e) {
FeedbackHelper.sendFeedback(SimImportService.this, TAG,
"Failed to import contacts from SIM card", e);
return false;
}
return true;
}
public SimCard getSim() {
return mSim;
}
@Override
protected void onPostExecute(Boolean success) {
super.onPostExecute(success);
stopSelf(mStartId);
Intent result;
final Notification notification;
if (success) {
result = new Intent(BROADCAST_SIM_IMPORT_COMPLETE)
.putExtra(EXTRA_RESULT_CODE, RESULT_SUCCESS)
.putExtra(EXTRA_RESULT_COUNT, mContacts.size())
.putExtra(EXTRA_OPERATION_REQUESTED_AT_TIME, mStartTime)
.putExtra(EXTRA_SIM_SUBSCRIPTION_ID, mSim.getSubscriptionId());
notification = getCompletedNotification();
} else {
result = new Intent(BROADCAST_SIM_IMPORT_COMPLETE)
.putExtra(EXTRA_RESULT_CODE, RESULT_FAILURE)
.putExtra(EXTRA_OPERATION_REQUESTED_AT_TIME, mStartTime)
.putExtra(EXTRA_SIM_SUBSCRIPTION_ID, mSim.getSubscriptionId());
notification = getFailedNotification();
}
LocalBroadcastManager.getInstance(SimImportService.this).sendBroadcast(result);
sPending.remove(this);
// Only notify of completion if all the import requests have finished. We're using
// the same notification for imports so in the rare case that a user has started
// multiple imports the notification won't go away until all of them complete.
if (sPending.isEmpty()) {
stopForeground(false);
mNotificationManager.notify(NOTIFICATION_ID, notification);
}
notifyStateChanged();
}
}
}