From fd6ef515b5c582c540b260c87cfb493bbd044643 Mon Sep 17 00:00:00 2001 From: Santos Cordon Date: Fri, 13 Feb 2015 00:29:13 -0800 Subject: Simple Backup Agent for the call log. Simple skeleton agent that reads the state of the call log. Does not actually do any backup. It logs what calls it would backup and what calls it would remove from backup. A subsequent CL will add the actual backup/restore of calls. Change-Id: Id8832c78a9a5aea71022b45c3cef79ca0b54f584 --- AndroidManifest.xml | 2 +- .../providers/contacts/CallLogBackupAgent.java | 215 +++++++++++++++++++++ .../contacts/DbModifierWithNotification.java | 5 + 3 files changed, 221 insertions(+), 1 deletion(-) create mode 100644 src/com/android/providers/contacts/CallLogBackupAgent.java diff --git a/AndroidManifest.xml b/AndroidManifest.xml index 20213d3..bd565b8 100644 --- a/AndroidManifest.xml +++ b/AndroidManifest.xml @@ -22,7 +22,7 @@ + android:backupAgent="CallLogBackupAgent"> callIds; + } + + private static class Call { + int id; + + @Override + public String toString() { + return "[" + id + "]"; + } + } + + private static final String TAG = "CallLogBackupAgent"; + private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG); + + /** Current version of CallLogBackup. Used to track the backup format. */ + private static final int VERSION = 1; + /** Version indicating that there exists no previous backup entry. */ + private static final int VERSION_NO_PREVIOUS_STATE = 0; + + private static final String[] CALL_LOG_PROJECTION = new String[] { + CallLog.Calls._ID, + CallLog.Calls.DATE, + CallLog.Calls.DURATION, + CallLog.Calls.NUMBER, + CallLog.Calls.TYPE, + CallLog.Calls.COUNTRY_ISO, + CallLog.Calls.GEOCODED_LOCATION, + CallLog.Calls.NUMBER_PRESENTATION, + CallLog.Calls.PHONE_ACCOUNT_COMPONENT_NAME, + CallLog.Calls.PHONE_ACCOUNT_ID, + CallLog.Calls.PHONE_ACCOUNT_ADDRESS + }; + + /** ${inheritDoc} */ + @Override + public void onBackup(ParcelFileDescriptor oldStateDescriptor, BackupDataOutput data, + ParcelFileDescriptor newStateDescriptor) throws IOException { + + // Get the list of the previous calls IDs which were backed up. + CallLogBackupState state = readState(oldStateDescriptor); + SortedSet callsToRemove = new TreeSet<>(state.callIds); + + // Get all the existing call log entries. + Cursor cursor = getAllCallLogEntries(); + if (DEBUG) { + Log.d(TAG, "Starting debug - state: " + state + ", cursor: " + cursor); + } + if (cursor == null) { + return; + } + + try { + // Loop through all the call log entries to identify: + // (1) new calls + // (2) calls which have been deleted. + while (cursor.moveToNext()) { + Call call = readCallFromCursor(cursor); + + if (!state.callIds.contains(call.id)) { + + if (DEBUG) { + Log.d(TAG, "Adding call to backup: " + call); + } + + // This call new (not in our list from the last backup), lets back it up. + addCallToBackup(data, call); + state.callIds.add(call.id); + } else { + // This call still exists in the current call log so delete it from the + // "callsToRemove" set since we want to keep it. + callsToRemove.remove(call.id); + } + } + + // Remove calls which no longer exist in the set. + for (Integer i : callsToRemove) { + if (DEBUG) { + Log.d(TAG, "Removing call from backup: " + i); + } + + removeCallFromBackup(data, i); + state.callIds.remove(i); + } + + // Rewrite the backup state. + writeState(newStateDescriptor, state); + } finally { + cursor.close(); + } + } + + /** ${inheritDoc} */ + @Override + public void onRestore(BackupDataInput data, int appVersionCode, ParcelFileDescriptor newState) + throws IOException { + if (DEBUG) { + Log.d(TAG, "Performing Restore"); + } + } + + private Cursor getAllCallLogEntries() { + // We use the API here instead of querying ContactsDatabaseHelper directly because + // CallLogProvider has special locks in place for sychronizing when to read. Using the APIs + // gives us that for free. + ContentResolver resolver = getContentResolver(); + return resolver.query(CallLog.Calls.CONTENT_URI, CALL_LOG_PROJECTION, null, null, null); + } + + private CallLogBackupState readState(ParcelFileDescriptor oldState) throws IOException { + DataInputStream dataInput = new DataInputStream( + new FileInputStream(oldState.getFileDescriptor())); + CallLogBackupState state = new CallLogBackupState(); + state.callIds = new TreeSet<>(); + + try { + // Read the version. + state.version = dataInput.readInt(); + + if (state.version >= 1) { + // Read the size. + int size = dataInput.readInt(); + + // Read all of the call IDs. + for (int i = 0; i < size; i++) { + state.callIds.add(dataInput.readInt()); + } + } + } catch (EOFException e) { + state.version = VERSION_NO_PREVIOUS_STATE; + } finally { + dataInput.close(); + } + + return state; + } + + private void writeState(ParcelFileDescriptor descriptor, CallLogBackupState state) + throws IOException { + DataOutputStream dataOutput = new DataOutputStream(new BufferedOutputStream( + new FileOutputStream(descriptor.getFileDescriptor()))); + + // Write version first of all + dataOutput.writeInt(VERSION); + + // [Version 1] + // size + callIds + dataOutput.writeInt(state.callIds.size()); + for (Integer i : state.callIds) { + dataOutput.writeInt(i); + } + + // Done! + dataOutput.close(); + } + + private Call readCallFromCursor(Cursor cursor) { + Call call = new Call(); + call.id = cursor.getInt(cursor.getColumnIndex(CallLog.Calls._ID)); + // TODO: Rest of call data. + return call; + } + + private void addCallToBackup(BackupDataOutput output, Call call) { + // TODO: Write the code + } + + private void removeCallFromBackup(BackupDataOutput output, int callId) { + // TODO: Write the code + } +} diff --git a/src/com/android/providers/contacts/DbModifierWithNotification.java b/src/com/android/providers/contacts/DbModifierWithNotification.java index 683a2bf..5ce41c4 100644 --- a/src/com/android/providers/contacts/DbModifierWithNotification.java +++ b/src/com/android/providers/contacts/DbModifierWithNotification.java @@ -20,6 +20,7 @@ package com.android.providers.contacts; import static android.Manifest.permission.ADD_VOICEMAIL; import static android.Manifest.permission.READ_VOICEMAIL; +import android.app.backup.BackupManager; import android.content.ComponentName; import android.content.ContentUris; import android.content.ContentValues; @@ -73,6 +74,8 @@ public class DbModifierWithNotification implements DatabaseModifier { private final boolean mIsCallsTable; private final VoicemailPermissions mVoicemailPermissions; + private BackupManager mBackupManager; + public DbModifierWithNotification(String tableName, SQLiteDatabase db, Context context) { this(tableName, db, null, context); } @@ -88,6 +91,7 @@ public class DbModifierWithNotification implements DatabaseModifier { mDb = db; mInsertHelper = insertHelper; mContext = context; + mBackupManager = new BackupManager(context); mBaseUri = mTableName.equals(Tables.VOICEMAIL_STATUS) ? Status.CONTENT_URI : Voicemails.CONTENT_URI; mIsCallsTable = mTableName.equals(Tables.CALLS); @@ -124,6 +128,7 @@ public class DbModifierWithNotification implements DatabaseModifier { private void notifyCallLogChange() { mContext.getContentResolver().notifyChange(Calls.CONTENT_URI, null, false); + mBackupManager.dataChanged(); } private void notifyVoicemailChangeOnInsert(Uri notificationUri, Set packagesModified) { -- cgit v1.1