aboutsummaryrefslogtreecommitdiffstats
path: root/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport
diff options
context:
space:
mode:
authorTim Bray <timbray@gmail.com>2014-09-12 08:34:51 -0700
committerTim Bray <timbray@gmail.com>2014-09-13 21:41:03 -0700
commit1c32d1df8801968ea6423ee4f1a9160e0c4ff0c3 (patch)
tree6571fc5b6aca0de6bd23e9acda38caf16d81cc30 /OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport
parent36ef8a9a62f32f0843c9e317d493a7f136624dce (diff)
downloadopen-keychain-1c32d1df8801968ea6423ee4f1a9160e0c4ff0c3.tar.gz
open-keychain-1c32d1df8801968ea6423ee4f1a9160e0c4ff0c3.tar.bz2
open-keychain-1c32d1df8801968ea6423ee4f1a9160e0c4ff0c3.zip
Add cloud search tab, lose Keybase/Keyserver tabs, re-organize prefs
Diffstat (limited to 'OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport')
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/CloudSearch.java90
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysList.java84
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysListEntry.java7
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/Keyserver.java16
4 files changed, 195 insertions, 2 deletions
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/CloudSearch.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/CloudSearch.java
new file mode 100644
index 000000000..27d826c59
--- /dev/null
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/CloudSearch.java
@@ -0,0 +1,90 @@
+/*
+ * Copyright (C) 2014 Tim Bray <tbray@textuality.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+package org.sufficientlysecure.keychain.keyimport;
+
+import org.sufficientlysecure.keychain.Constants;
+import org.sufficientlysecure.keychain.helper.Preferences;
+import org.sufficientlysecure.keychain.util.Log;
+
+import java.util.ArrayList;
+import java.util.Vector;
+
+/**
+ * Search two or more types of server for online keys.
+ */
+public class CloudSearch {
+
+ private final static long SECONDS = 1000;
+
+ public static ArrayList<ImportKeysListEntry> search(final String query, Preferences.CloudSearchPrefs cloudPrefs)
+ throws Keyserver.CloudSearchFailureException {
+ final ArrayList<Keyserver> servers = new ArrayList<Keyserver>();
+
+ // it's a Vector for sync, multiple threads might report problems
+ final Vector<Keyserver.CloudSearchFailureException> problems = new Vector<Keyserver.CloudSearchFailureException>();
+
+ if (cloudPrefs.searchKeyserver) {
+ servers.add(new HkpKeyserver(cloudPrefs.keyserver));
+ }
+ if (cloudPrefs.searchKeybase) {
+ servers.add(new KeybaseKeyserver());
+ }
+ final ImportKeysList results = new ImportKeysList(servers.size());
+
+ for (final Keyserver keyserver : servers) {
+ Runnable r = new Runnable() {
+ @Override
+ public void run() {
+ try {
+ results.addAll(keyserver.search(query));
+ } catch (Keyserver.CloudSearchFailureException e) {
+ problems.add(e);
+ }
+ results.finishedAdding(); // notifies if all searchers done
+ }
+ };
+ new Thread(r).start();
+ }
+
+ // wait for either all the searches to come back, or 10 seconds
+ synchronized(results) {
+ try {
+ results.wait(10 * SECONDS);
+ } catch (InterruptedException e) {
+ }
+ }
+
+ if (results.outstandingSuppliers() > 0) {
+ String message = "Launched " + servers.size() + " cloud searchers, but" +
+ results.outstandingSuppliers() + "failed to complete.";
+ problems.add(new Keyserver.QueryFailedException(message));
+ }
+
+ if (!problems.isEmpty()) {
+ for (Keyserver.CloudSearchFailureException e : problems) {
+ Log.d(Constants.TAG, "Cloud search exception: " + e.getLocalizedMessage());
+ }
+
+ // only throw exception if we didn’t get any results
+ if (results.isEmpty()) {
+ throw problems.get(0);
+ }
+ }
+
+ return results;
+ }
+}
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysList.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysList.java
new file mode 100644
index 000000000..15de8cb8f
--- /dev/null
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysList.java
@@ -0,0 +1,84 @@
+/*
+ * Copyright (C) 2014 Tim Bray <tbray@textuality.com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+package org.sufficientlysecure.keychain.keyimport;
+
+import java.util.ArrayList;
+import java.util.Collection;
+
+/**
+ * Just an ArrayList, only with a synchronized dupe-merging add/addAll, and a sign-off method
+ */
+public class ImportKeysList extends ArrayList<ImportKeysListEntry> {
+
+ private int mSupplierCount;
+
+ public ImportKeysList(int supplierCount) {
+ mSupplierCount = supplierCount;
+ }
+
+ @Override
+ public boolean add(ImportKeysListEntry toAdd) {
+ addOrMerge(toAdd);
+ return true; // that’s what the ArrayList#add contract says
+ }
+
+ @Override
+ public boolean addAll(Collection<? extends ImportKeysListEntry> addThese) {
+ boolean modified = false;
+ for (ImportKeysListEntry toAdd : addThese) {
+ modified = addOrMerge(toAdd) || modified;
+ }
+ return modified;
+ }
+
+ // NOTE: side-effects
+ // NOTE: synchronized
+ private synchronized boolean addOrMerge(ImportKeysListEntry toAdd) {
+ for (ImportKeysListEntry existing : this) {
+ if (toAdd.hasSameKeyAs(existing)) {
+ return mergeIDs(toAdd, existing);
+ }
+ }
+ return super.add(toAdd);
+ }
+
+ // being a little anal about the ArrayList#addAll contract here
+ private boolean mergeIDs(ImportKeysListEntry incoming, ImportKeysListEntry existing) {
+ boolean modified = false;
+ ArrayList<String> incomingIDs = incoming.getUserIds();
+ ArrayList<String> existingIDs = existing.getUserIds();
+ for (String incomingID : incomingIDs) {
+ if (!existingIDs.contains(incomingID)) {
+ existingIDs.add(incomingID);
+ modified = true;
+ }
+ }
+ return modified;
+ }
+
+ // NOTE: synchronized
+ public synchronized void finishedAdding() {
+ mSupplierCount--;
+ if (mSupplierCount == 0) {
+ this.notify();
+ }
+ }
+
+ public int outstandingSuppliers() {
+ return mSupplierCount;
+ }
+}
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysListEntry.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysListEntry.java
index da70f1505..c10cd2e70 100644
--- a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysListEntry.java
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/ImportKeysListEntry.java
@@ -107,6 +107,13 @@ public class ImportKeysListEntry implements Serializable, Parcelable {
return super.hashCode();
}
+ public boolean hasSameKeyAs(Object o) {
+ if (mFingerprintHex == null || o == null || !(o instanceof ImportKeysListEntry)) {
+ return false;
+ }
+ return mFingerprintHex.equals(((ImportKeysListEntry) o).mFingerprintHex);
+ }
+
public String getKeyIdHex() {
return mKeyIdHex;
}
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/Keyserver.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/Keyserver.java
index b726529f8..5e4bd0b70 100644
--- a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/Keyserver.java
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/keyimport/Keyserver.java
@@ -24,7 +24,19 @@ import java.io.InputStream;
import java.util.List;
public abstract class Keyserver {
- public static class QueryFailedException extends Exception {
+
+ public static class CloudSearchFailureException extends Exception {
+ private static final long serialVersionUID = 2703768928624654515L;
+
+ public CloudSearchFailureException(String message) {
+ super(message);
+ }
+ public CloudSearchFailureException() {
+ super();
+ }
+ }
+
+ public static class QueryFailedException extends CloudSearchFailureException {
private static final long serialVersionUID = 2703768928624654512L;
public QueryFailedException(String message) {
@@ -32,7 +44,7 @@ public abstract class Keyserver {
}
}
- public static class QueryNeedsRepairException extends Exception {
+ public static class QueryNeedsRepairException extends CloudSearchFailureException {
private static final long serialVersionUID = 2693768928624654512L;
}