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;
|
2013-03-14 03:23:50 +00:00
|
|
|
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;
|
2012-09-11 19:56:54 +02:00
|
|
|
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;
|
2012-09-11 19:56:54 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
import android.view.KeyEvent;
|
2012-06-09 18:04:18 +03:00
|
|
|
import android.view.LayoutInflater;
|
|
|
|
|
import android.view.View;
|
2012-09-11 19:56:54 +02:00
|
|
|
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;
|
2012-09-11 19:56:54 +02:00
|
|
|
import android.widget.TextView;
|
|
|
|
|
import android.widget.TextView.OnEditorActionListener;
|
2012-06-09 18:04:18 +03:00
|
|
|
import android.widget.Toast;
|
|
|
|
|
|
2012-09-11 19:56:54 +02:00
|
|
|
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;
|
|
|
|
|
|
2013-01-06 18:13:46 +01:00
|
|
|
private Messenger mMessenger;
|
2012-09-11 19:56:54 +02:00
|
|
|
private EditText mPassphraseEditText;
|
2013-03-14 03:23:50 +00:00
|
|
|
private boolean canKB;
|
2012-09-11 19:56:54 +02:00
|
|
|
|
2012-06-09 18:04:18 +03:00
|
|
|
/**
|
2012-06-18 19:29:23 +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
|
|
|
*/
|
2012-09-11 19:56:54 +02: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)) {
|
2012-09-11 19:56:54 +02:00
|
|
|
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
|
|
|
|
|
*/
|
2012-09-11 19:56:54 +02:00
|
|
|
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
|
2012-11-14 16:02:11 +01:00
|
|
|
.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
|
2012-09-11 19:56:54 +02:00
|
|
|
PassphraseCacheService.addCachedPassphrase(context, secretKey.getKeyID(), "");
|
2012-06-09 19:12:19 +03:00
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
} catch (PGPException e) {
|
|
|
|
|
// silently catch
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2012-09-11 19:56:54 +02:00
|
|
|
@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();
|
2013-03-14 03:23:50 +00:00
|
|
|
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???
|
2013-03-14 03:23:50 +00:00
|
|
|
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);
|
2013-03-14 03:23:50 +00:00
|
|
|
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);
|
|
|
|
|
|
2012-09-11 19:56:54 +02:00
|
|
|
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() {
|
2013-01-06 18:13:46 +01:00
|
|
|
|
|
|
|
|
@Override
|
2012-06-09 18:04:18 +03:00
|
|
|
public void onClick(DialogInterface dialog, int id) {
|
|
|
|
|
dismiss();
|
2013-03-14 12:28:22 +00:00
|
|
|
long curKeyIndex = 1;
|
2013-03-14 03:23:50 +00:00
|
|
|
boolean keyOK = true;
|
2012-09-11 19:56:54 +02:00
|
|
|
String passPhrase = mPassphraseEditText.getText().toString();
|
2012-06-09 18:04:18 +03:00
|
|
|
long keyId;
|
2013-03-14 03:23:50 +00:00
|
|
|
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();
|
2013-03-14 03:23:50 +00:00
|
|
|
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");
|
2012-09-11 19:56:54 +02:00
|
|
|
PassphraseCacheService.addCachedPassphrase(activity, keyId, passPhrase);
|
2013-03-14 03:23:50 +00:00
|
|
|
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() {
|
2013-01-06 18:13:46 +01:00
|
|
|
|
|
|
|
|
@Override
|
2012-06-09 18:04:18 +03:00
|
|
|
public void onClick(DialogInterface dialog, int id) {
|
|
|
|
|
dismiss();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2013-03-14 03:23:50 +00:00
|
|
|
canKB = true;
|
2012-06-09 18:04:18 +03:00
|
|
|
return alert.create();
|
|
|
|
|
}
|
|
|
|
|
|
2012-09-11 19:56:54 +02:00
|
|
|
@Override
|
|
|
|
|
public void onActivityCreated(Bundle arg0) {
|
|
|
|
|
super.onActivityCreated(arg0);
|
2013-03-14 03:23:50 +00:00
|
|
|
if (canKB) {
|
2012-09-11 19:56:54 +02:00
|
|
|
// request focus and open soft keyboard
|
2013-03-14 03:23:50 +00:00
|
|
|
mPassphraseEditText.requestFocus();
|
|
|
|
|
getDialog().getWindow().setSoftInputMode(LayoutParams.SOFT_INPUT_STATE_VISIBLE);
|
2012-09-11 19:56:54 +02:00
|
|
|
|
2013-03-14 03:23:50 +00:00
|
|
|
mPassphraseEditText.setOnEditorActionListener(this);
|
|
|
|
|
}
|
2012-09-11 19:56:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 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);
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-09-11 19:56:54 +02:00
|
|
|
|
2013-03-14 03:23:50 +00:00
|
|
|
}
|
|
|
|
|
|