Files
open-keychain/OpenPGP-Keychain/src/org/sufficientlysecure/keychain/ui/dialog/PassphraseDialogFragment.java

302 lines
12 KiB
Java
Raw Normal View History

2012-06-09 18:04:18 +03:00
/*
* Copyright (C) 2012 Dominik Schürmann <dominik@dominikschuermann.de>
*
* 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.
*/
2013-01-16 14:31:16 +01:00
package org.sufficientlysecure.keychain.ui.dialog;
2012-06-09 18:04:18 +03:00
import org.spongycastle.openpgp.PGPException;
import org.spongycastle.openpgp.PGPPrivateKey;
import org.spongycastle.openpgp.PGPSecretKey;
import org.spongycastle.openpgp.PGPSecretKeyRing;
2012-06-09 19:12:19 +03:00
import org.spongycastle.openpgp.operator.PBESecretKeyDecryptor;
import org.spongycastle.openpgp.operator.jcajce.JcePBESecretKeyDecryptorBuilder;
2013-01-16 14:31:16 +01:00
import org.sufficientlysecure.keychain.Constants;
import org.sufficientlysecure.keychain.Id;
import org.sufficientlysecure.keychain.helper.PgpHelper;
import org.sufficientlysecure.keychain.helper.PgpMain;
import org.sufficientlysecure.keychain.helper.PgpMain.PgpGeneralException;
import org.sufficientlysecure.keychain.provider.ProviderHelper;
import org.sufficientlysecure.keychain.service.PassphraseCacheService;
import org.sufficientlysecure.keychain.util.Log;
import org.sufficientlysecure.keychain.R;
2012-06-09 18:04:18 +03:00
import android.app.Activity;
import android.app.AlertDialog;
import android.app.Dialog;
import android.content.Context;
2012-06-09 18:04:18 +03:00
import android.content.DialogInterface;
import android.content.DialogInterface.OnClickListener;
import android.os.Bundle;
import android.os.Message;
import android.os.Messenger;
import android.os.RemoteException;
import android.support.v4.app.DialogFragment;
import android.view.KeyEvent;
2012-06-09 18:04:18 +03:00
import android.view.LayoutInflater;
import android.view.View;
import android.view.WindowManager.LayoutParams;
import android.view.inputmethod.EditorInfo;
import android.widget.Button;
2012-06-09 18:04:18 +03:00
import android.widget.EditText;
import android.widget.TextView;
import android.widget.TextView.OnEditorActionListener;
2012-06-09 18:04:18 +03:00
import android.widget.Toast;
public class PassphraseDialogFragment extends DialogFragment implements OnEditorActionListener {
2012-06-09 18:04:18 +03:00
private static final String ARG_MESSENGER = "messenger";
private static final String ARG_SECRET_KEY_ID = "secret_key_id";
public static final int MESSAGE_OKAY = 1;
private Messenger mMessenger;
private EditText mPassphraseEditText;
private boolean canKB;
2012-06-09 18:04:18 +03:00
/**
* Creates new instance of this dialog fragment
2012-06-09 18:04:18 +03:00
*
* @param secretKeyId
* secret key id you want to use
* @param messenger
* to communicate back after caching the passphrase
* @return
2013-01-16 14:31:16 +01:00
* @throws PgpGeneralException
2012-06-09 18:04:18 +03:00
*/
public static PassphraseDialogFragment newInstance(Context context, Messenger messenger,
2013-01-16 14:31:16 +01:00
long secretKeyId) throws PgpGeneralException {
2012-06-09 19:12:19 +03:00
// check if secret key has a passphrase
if (!(secretKeyId == Id.key.symmetric || secretKeyId == Id.key.none)) {
if (!hasPassphrase(context, secretKeyId)) {
2013-01-16 14:31:16 +01:00
throw new PgpMain.PgpGeneralException("No passphrase! No passphrase dialog needed!");
2012-06-09 19:12:19 +03:00
}
}
2012-06-09 18:04:18 +03:00
PassphraseDialogFragment frag = new PassphraseDialogFragment();
Bundle args = new Bundle();
args.putLong(ARG_SECRET_KEY_ID, secretKeyId);
args.putParcelable(ARG_MESSENGER, messenger);
frag.setArguments(args);
2012-06-09 19:12:19 +03:00
2012-06-09 18:04:18 +03:00
return frag;
}
2012-06-09 19:12:19 +03:00
/**
* Checks if key has a passphrase
*
* @param secretKeyId
* @return true if it has a passphrase
*/
private static boolean hasPassphrase(Context context, long secretKeyId) {
2012-06-09 19:12:19 +03:00
// check if the key has no passphrase
try {
2013-01-16 14:31:16 +01:00
PGPSecretKey secretKey = PgpHelper.getMasterKey(ProviderHelper
.getPGPSecretKeyRingByKeyId(context, secretKeyId));
2012-10-25 14:52:13 +02:00
// PGPSecretKey secretKey =
// PGPHelper.getMasterKey(PGPMain.getSecretKeyRing(secretKeyId));
2012-06-09 19:12:19 +03:00
Log.d(Constants.TAG, "Check if key has no passphrase...");
PBESecretKeyDecryptor keyDecryptor = new JcePBESecretKeyDecryptorBuilder().setProvider(
"SC").build("".toCharArray());
PGPPrivateKey testKey = secretKey.extractPrivateKey(keyDecryptor);
if (testKey != null) {
Log.d(Constants.TAG, "Key has no passphrase! Caches empty passphrase!");
// cache empty passphrase
PassphraseCacheService.addCachedPassphrase(context, secretKey.getKeyID(), "");
2012-06-09 19:12:19 +03:00
return false;
}
} catch (PGPException e) {
// silently catch
}
return true;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
2012-06-09 18:04:18 +03:00
/**
* Creates dialog
*/
@Override
public Dialog onCreateDialog(Bundle savedInstanceState) {
final Activity activity = getActivity();
final long secretKeyId = getArguments().getLong(ARG_SECRET_KEY_ID);
2012-06-09 18:04:18 +03:00
mMessenger = getArguments().getParcelable(ARG_MESSENGER);
AlertDialog.Builder alert = new AlertDialog.Builder(activity);
alert.setTitle(R.string.title_authentication);
final PGPSecretKey secretKey;
if (secretKeyId == Id.key.symmetric || secretKeyId == Id.key.none) {
secretKey = null;
2012-09-11 15:27:32 +02:00
alert.setMessage(R.string.passPhraseForSymmetricEncryption);
2012-06-09 18:04:18 +03:00
} else {
2012-10-25 14:52:13 +02:00
// TODO: by master key id???
secretKey = PgpHelper.getMasterKey(ProviderHelper.getPGPSecretKeyRingByKeyId(activity, secretKeyId));
2012-10-25 14:52:13 +02:00
// secretKey = PGPHelper.getMasterKey(PGPMain.getSecretKeyRing(secretKeyId));
2012-06-09 18:04:18 +03:00
if (secretKey == null) {
alert.setTitle(R.string.title_keyNotFound);
alert.setMessage(getString(R.string.keyNotFound, secretKeyId));
alert.setPositiveButton(android.R.string.ok, new OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
dismiss();
}
});
alert.setCancelable(false);
canKB = false;
2012-06-09 18:04:18 +03:00
return alert.create();
}
2013-01-16 14:31:16 +01:00
String userId = PgpHelper.getMainUserIdSafe(activity, secretKey);
2012-09-11 15:27:32 +02:00
Log.d(Constants.TAG, "User id: '" + userId + "'");
2012-06-09 18:04:18 +03:00
alert.setMessage(getString(R.string.passPhraseFor, userId));
}
LayoutInflater inflater = activity.getLayoutInflater();
View view = inflater.inflate(R.layout.passphrase, null);
alert.setView(view);
mPassphraseEditText = (EditText) view.findViewById(R.id.passphrase_passphrase);
2012-09-11 15:27:32 +02:00
2012-06-09 18:04:18 +03:00
alert.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
@Override
2012-06-09 18:04:18 +03:00
public void onClick(DialogInterface dialog, int id) {
dismiss();
long curKeyIndex = 1;
boolean keyOK = true;
String passPhrase = mPassphraseEditText.getText().toString();
2012-06-09 18:04:18 +03:00
long keyId;
PGPSecretKey clickSecretKey = secretKey;
if (clickSecretKey != null) {
while (keyOK == true) {
if (clickSecretKey != null) { //check again for loop
try {
PBESecretKeyDecryptor keyDecryptor = new JcePBESecretKeyDecryptorBuilder()
.setProvider(PgpMain.BOUNCY_CASTLE_PROVIDER_NAME).build(
passPhrase.toCharArray());
PGPPrivateKey testKey = clickSecretKey.extractPrivateKey(keyDecryptor);
if (testKey == null) {
if (!clickSecretKey.isMasterKey()) {
Toast.makeText(activity, R.string.error_couldNotExtractPrivateKey,
Toast.LENGTH_SHORT).show();
return;
} else {
clickSecretKey = PgpHelper.getKeyNum(ProviderHelper.getPGPSecretKeyRingByKeyId(activity, secretKeyId), curKeyIndex);
curKeyIndex++; //does post-increment work like C?
continue;
}
} else {
keyOK = false;
}
} catch (PGPException e) {
Toast.makeText(activity, R.string.wrongPassPhrase, Toast.LENGTH_SHORT)
.show();
return;
}
} else {
2012-06-09 18:04:18 +03:00
Toast.makeText(activity, R.string.error_couldNotExtractPrivateKey,
Toast.LENGTH_SHORT).show();
return; //ran out of keys to try
2012-06-09 18:04:18 +03:00
}
}
keyId = secretKey.getKeyID();
} else {
keyId = Id.key.symmetric;
}
2012-06-09 19:12:19 +03:00
// cache the new passphrase
Log.d(Constants.TAG, "Everything okay! Caching entered passphrase");
PassphraseCacheService.addCachedPassphrase(activity, keyId, passPhrase);
if (keyOK == false) {
PassphraseCacheService.addCachedPassphrase(activity, clickSecretKey.getKeyID(), passPhrase);
}
2012-06-09 19:12:19 +03:00
2012-06-09 18:04:18 +03:00
sendMessageToHandler(MESSAGE_OKAY);
}
});
alert.setNegativeButton(android.R.string.cancel, new DialogInterface.OnClickListener() {
@Override
2012-06-09 18:04:18 +03:00
public void onClick(DialogInterface dialog, int id) {
dismiss();
}
});
canKB = true;
2012-06-09 18:04:18 +03:00
return alert.create();
}
@Override
public void onActivityCreated(Bundle arg0) {
super.onActivityCreated(arg0);
if (canKB) {
// request focus and open soft keyboard
mPassphraseEditText.requestFocus();
getDialog().getWindow().setSoftInputMode(LayoutParams.SOFT_INPUT_STATE_VISIBLE);
mPassphraseEditText.setOnEditorActionListener(this);
}
}
/**
* Associate the "done" button on the soft keyboard with the okay button in the view
*/
@Override
public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
if (EditorInfo.IME_ACTION_DONE == actionId) {
AlertDialog dialog = ((AlertDialog) getDialog());
Button bt = dialog.getButton(AlertDialog.BUTTON_POSITIVE);
bt.performClick();
return true;
}
return false;
}
2012-06-09 18:04:18 +03:00
/**
* Send message back to handler which is initialized in a activity
*
2012-06-09 19:12:19 +03:00
* @param what
* Message integer you want to send
2012-06-09 18:04:18 +03:00
*/
2012-06-09 19:12:19 +03:00
private void sendMessageToHandler(Integer what) {
2012-06-09 18:04:18 +03:00
Message msg = Message.obtain();
2012-06-09 19:12:19 +03:00
msg.what = what;
2012-06-09 18:04:18 +03:00
try {
mMessenger.send(msg);
} catch (RemoteException e) {
Log.w(Constants.TAG, "Exception sending message, Is handler present?", e);
} catch (NullPointerException e) {
Log.w(Constants.TAG, "Messenger is null!", e);
}
}
}