diff options
Diffstat (limited to 'OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/AddKeysActivity.java')
-rw-r--r-- | OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/AddKeysActivity.java | 296 |
1 files changed, 286 insertions, 10 deletions
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/AddKeysActivity.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/AddKeysActivity.java index 05d6c2779..ecc72c24d 100644 --- a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/AddKeysActivity.java +++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/AddKeysActivity.java @@ -17,33 +17,52 @@ package org.sufficientlysecure.keychain.ui; +import android.app.ProgressDialog; import android.content.Intent; import android.graphics.PorterDuff; import android.net.Uri; import android.os.Bundle; +import android.os.Message; +import android.os.Messenger; +import android.support.v4.app.LoaderManager; +import android.support.v4.content.Loader; +import android.support.v4.util.LongSparseArray; import android.support.v7.app.ActionBarActivity; import android.view.View; import android.widget.ImageView; import org.sufficientlysecure.keychain.Constants; import org.sufficientlysecure.keychain.R; -import org.sufficientlysecure.keychain.pgp.CanonicalizedPublicKeyRing; -import org.sufficientlysecure.keychain.provider.CachedPublicKeyRing; +import org.sufficientlysecure.keychain.keyimport.ImportKeysListEntry; +import org.sufficientlysecure.keychain.keyimport.Keyserver; +import org.sufficientlysecure.keychain.keyimport.ParcelableKeyRing; import org.sufficientlysecure.keychain.provider.KeychainContract; import org.sufficientlysecure.keychain.provider.ProviderHelper; +import org.sufficientlysecure.keychain.service.KeychainIntentService; +import org.sufficientlysecure.keychain.service.KeychainIntentServiceHandler; +import org.sufficientlysecure.keychain.service.results.ImportKeyResult; import org.sufficientlysecure.keychain.service.results.OperationResult; +import org.sufficientlysecure.keychain.ui.adapter.AsyncTaskResultWrapper; +import org.sufficientlysecure.keychain.ui.adapter.ImportKeysListCloudLoader; +import org.sufficientlysecure.keychain.ui.adapter.ImportKeysListLoader; import org.sufficientlysecure.keychain.ui.util.Notify; import org.sufficientlysecure.keychain.ui.widget.ExchangeKeySpinner; import org.sufficientlysecure.keychain.ui.widget.KeySpinner; +import org.sufficientlysecure.keychain.util.InputData; import org.sufficientlysecure.keychain.util.Log; +import org.sufficientlysecure.keychain.util.ParcelableFileCache; +import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; +import java.util.ArrayList; +import java.util.Iterator; import edu.cmu.cylab.starslinger.exchange.ExchangeActivity; import edu.cmu.cylab.starslinger.exchange.ExchangeConfig; -public class AddKeysActivity extends ActionBarActivity { +public class AddKeysActivity extends ActionBarActivity implements + LoaderManager.LoaderCallbacks<AsyncTaskResultWrapper<ArrayList<ImportKeysListEntry>>> { ExchangeKeySpinner mSafeSlingerKeySpinner; View mActionSafeSlinger; @@ -55,9 +74,16 @@ public class AddKeysActivity extends ActionBarActivity { long mExchangeMasterKeyId = Constants.key.none; - private static final int REQUEST_CODE_SAFESLINGER = 1; + byte[] mImportBytes; + private LongSparseArray<ParcelableKeyRing> mCachedKeyData; + private static final int REQUEST_CODE_SAFE_SLINGER = 1; + + + private static final int LOADER_ID_BYTES = 0; + private static final int LOADER_ID_CLOUD = 1; + @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); @@ -119,7 +145,7 @@ public class AddKeysActivity extends ActionBarActivity { Intent slingerIntent = new Intent(this, ExchangeActivity.class); slingerIntent.putExtra(ExchangeConfig.extra.USER_DATA, keyBlob); slingerIntent.putExtra(ExchangeConfig.extra.HOST_NAME, Constants.SAFESLINGER_SERVER); - startActivityForResult(slingerIntent, REQUEST_CODE_SAFESLINGER); + startActivityForResult(slingerIntent, REQUEST_CODE_SAFE_SLINGER); } catch (ProviderHelper.NotFoundException e) { Log.e(Constants.TAG, "personal key not found", e); } @@ -143,14 +169,16 @@ public class AddKeysActivity extends ActionBarActivity { result.createNotify(this).show(); } else { switch (requestCode) { - case REQUEST_CODE_SAFESLINGER: + case REQUEST_CODE_SAFE_SLINGER: switch (resultCode) { case ExchangeActivity.RESULT_EXCHANGE_OK: // import exchanged keys - Intent importIntent = new Intent(this, ImportKeysActivity.class); - importIntent.setAction(ImportKeysActivity.ACTION_IMPORT_KEY); - importIntent.putExtra(ImportKeysActivity.EXTRA_KEY_BYTES, getSlingedKeys(data)); - startActivity(importIntent); + mImportBytes = getSlingedKeys(data); + getSupportLoaderManager().restartLoader(LOADER_ID_BYTES, null, this); +// Intent importIntent = new Intent(this, ImportKeysActivity.class); +// importIntent.setAction(ImportKeysActivity.ACTION_IMPORT_KEY); +// importIntent.putExtra(ImportKeysActivity.EXTRA_KEY_BYTES, getSlingedKeys(data)); +// startActivity(importIntent); break; case ExchangeActivity.RESULT_EXCHANGE_CANCELED: // handle canceled result @@ -185,4 +213,252 @@ public class AddKeysActivity extends ActionBarActivity { return out.toByteArray(); } + + @Override + public Loader<AsyncTaskResultWrapper<ArrayList<ImportKeysListEntry>>> onCreateLoader(int id, Bundle args) { + switch (id) { + case LOADER_ID_BYTES: { + InputData inputData = new InputData(new ByteArrayInputStream(mImportBytes), mImportBytes.length); + return new ImportKeysListLoader(this, inputData); + } + case LOADER_ID_CLOUD: { +// ImportKeysListFragment.CloudLoaderState ls = (ImportKeysListFragment.CloudLoaderState) mLoaderState; +// return new ImportKeysListCloudLoader(this, ls.mServerQuery, ls.mCloudPrefs); + } + + default: + return null; + } + } + + @Override + public void onLoadFinished(Loader<AsyncTaskResultWrapper<ArrayList<ImportKeysListEntry>>> loader, AsyncTaskResultWrapper<ArrayList<ImportKeysListEntry>> data) { + + Log.d(Constants.TAG, "data: " + data.getResult()); + + // swap in the real data! +// mAdapter.setData(data.getResult()); +// mAdapter.notifyDataSetChanged(); +// +// setListAdapter(mAdapter); +// +// // The list should now be shown. +// if (isResumed()) { +// setListShown(true); +// } else { +// setListShownNoAnimation(true); +// } + + Exception error = data.getError(); + + // free old cached key data + mCachedKeyData = null; + + + // TODO: Use parcels!!!!!!!!!!!!!!! + switch (loader.getId()) { + case LOADER_ID_BYTES: + + if (error == null) { + // No error + mCachedKeyData = ((ImportKeysListLoader) loader).getParcelableRings(); + } else if (error instanceof ImportKeysListLoader.NoValidKeysException) { + Notify.showNotify(this, R.string.error_import_no_valid_keys, Notify.Style.ERROR); + } else if (error instanceof ImportKeysListLoader.NonPgpPartException) { + Notify.showNotify(this, + ((ImportKeysListLoader.NonPgpPartException) error).getCount() + " " + getResources(). + getQuantityString(R.plurals.error_import_non_pgp_part, + ((ImportKeysListLoader.NonPgpPartException) error).getCount()), + Notify.Style.OK + ); + } else { + Notify.showNotify(this, R.string.error_generic_report_bug, Notify.Style.ERROR); + } + break; + + case LOADER_ID_CLOUD: + + if (error == null) { + // No error + } else if (error instanceof Keyserver.QueryTooShortException) { + Notify.showNotify(this, R.string.error_query_too_short, Notify.Style.ERROR); + } else if (error instanceof Keyserver.TooManyResponsesException) { + Notify.showNotify(this, R.string.error_too_many_responses, Notify.Style.ERROR); + } else if (error instanceof Keyserver.QueryTooShortOrTooManyResponsesException) { + Notify.showNotify(this, R.string.error_too_short_or_too_many_responses, Notify.Style.ERROR); + } else if (error instanceof Keyserver.QueryFailedException) { + Log.d(Constants.TAG, + "Unrecoverable keyserver query error: " + error.getLocalizedMessage()); + String alert = this.getString(R.string.error_searching_keys); + alert = alert + " (" + error.getLocalizedMessage() + ")"; + Notify.showNotify(this, alert, Notify.Style.ERROR); + } + break; + + default: + break; + } + + importKeys(); + } + + @Override + public void onLoaderReset(Loader<AsyncTaskResultWrapper<ArrayList<ImportKeysListEntry>>> loader) { + switch (loader.getId()) { + case LOADER_ID_BYTES: + // Clear the data in the adapter. +// mAdapter.clear(); + break; + case LOADER_ID_CLOUD: + // Clear the data in the adapter. +// mAdapter.clear(); + break; + default: + break; + } + } + + public ParcelableFileCache.IteratorWithSize<ParcelableKeyRing> getSelectedData() { + return new ParcelableFileCache.IteratorWithSize<ParcelableKeyRing>() { + int i = 0; + + @Override + public int getSize() { + return mCachedKeyData.size(); + } + + @Override + public boolean hasNext() { + return (mCachedKeyData.get(i + 1) != null); + } + + @Override + public ParcelableKeyRing next() { + ParcelableKeyRing key = mCachedKeyData.get(i); + i++; + return key; + } + + @Override + public void remove() { + mCachedKeyData.remove(i); + } + }; + } + + public void importKeys() { + // Message is received after importing is done in KeychainIntentService + KeychainIntentServiceHandler saveHandler = new KeychainIntentServiceHandler( + this, + getString(R.string.progress_importing), + ProgressDialog.STYLE_HORIZONTAL, + true) { + public void handleMessage(Message message) { + // handle messages by standard KeychainIntentServiceHandler first + super.handleMessage(message); + + if (message.arg1 == KeychainIntentServiceHandler.MESSAGE_OKAY) { + // get returned data bundle + Bundle returnData = message.getData(); + if (returnData == null) { + return; + } + final ImportKeyResult result = + returnData.getParcelable(OperationResult.EXTRA_RESULT); + if (result == null) { + Log.e(Constants.TAG, "result == null"); + return; + } + +// if (ACTION_IMPORT_KEY_FROM_KEYSERVER_AND_RETURN_RESULT.equals(getIntent().getAction()) +// || ACTION_IMPORT_KEY_FROM_FILE_AND_RETURN.equals(getIntent().getAction())) { +// Intent intent = new Intent(); +// intent.putExtra(ImportKeyResult.EXTRA_RESULT, result); +// ImportKeysActivity.this.setResult(RESULT_OK, intent); +// ImportKeysActivity.this.finish(); +// return; +// } +// if (ACTION_IMPORT_KEY_FROM_KEYSERVER_AND_RETURN_TO_SERVICE.equals(getIntent().getAction())) { +// ImportKeysActivity.this.setResult(RESULT_OK, mPendingIntentData); +// ImportKeysActivity.this.finish(); +// return; +// } + + result.createNotify(AddKeysActivity.this).show(); + } + } + }; + +// ImportKeysListFragment.LoaderState ls = mListFragment.getLoaderState(); +// if (importBytes != null) { + Log.d(Constants.TAG, "importKeys started"); + + // Send all information needed to service to import key in other thread + Intent intent = new Intent(this, KeychainIntentService.class); + + intent.setAction(KeychainIntentService.ACTION_IMPORT_KEYRING); + + // fill values for this action + Bundle data = new Bundle(); + + // get DATA from selected key entries +// ParcelableFileCache.IteratorWithSize<ParcelableKeyRing> selectedEntries = mListFragment.getSelectedData(); + + // instead of giving the entries by Intent extra, cache them into a + // file to prevent Java Binder problems on heavy imports + // read FileImportCache for more info. + try { + // We parcel this iteratively into a file - anything we can + // display here, we should be able to import. + ParcelableFileCache<ParcelableKeyRing> cache = + new ParcelableFileCache<ParcelableKeyRing>(this, "key_import.pcl"); + cache.writeCache(getSelectedData()); + + intent.putExtra(KeychainIntentService.EXTRA_DATA, data); + + // Create a new Messenger for the communication back + Messenger messenger = new Messenger(saveHandler); + intent.putExtra(KeychainIntentService.EXTRA_MESSENGER, messenger); + + // show progress dialog + saveHandler.showProgressDialog(this); + + // start service with intent + startService(intent); + } catch (IOException e) { + Log.e(Constants.TAG, "Problem writing cache file", e); + Notify.showNotify(this, "Problem writing cache file!", Notify.Style.ERROR); + } +// } else if (ls instanceof ImportKeysListFragment.CloudLoaderState) { +// ImportKeysListFragment.CloudLoaderState sls = (ImportKeysListFragment.CloudLoaderState) ls; +// +// // Send all information needed to service to query keys in other thread +// Intent intent = new Intent(this, KeychainIntentService.class); +// +// intent.setAction(KeychainIntentService.ACTION_DOWNLOAD_AND_IMPORT_KEYS); +// +// // fill values for this action +// Bundle data = new Bundle(); +// +// data.putString(KeychainIntentService.DOWNLOAD_KEY_SERVER, sls.mCloudPrefs.keyserver); +// +// // get selected key entries +// ArrayList<ImportKeysListEntry> selectedEntries = mListFragment.getSelectedEntries(); +// data.putParcelableArrayList(KeychainIntentService.DOWNLOAD_KEY_LIST, selectedEntries); +// +// intent.putExtra(KeychainIntentService.EXTRA_DATA, data); +// +// // Create a new Messenger for the communication back +// Messenger messenger = new Messenger(saveHandler); +// intent.putExtra(KeychainIntentService.EXTRA_MESSENGER, messenger); +// +// // show progress dialog +// saveHandler.showProgressDialog(this); +// +// // start service with intent +// startService(intent); +// } else { +// Notify.showNotify(this, R.string.error_nothing_import, Notify.Style.ERROR); +// } + } } |