extract UpdatedKeys access from KeychainProvider into KeyMetadataDao
This commit is contained in:
@@ -26,8 +26,8 @@ import org.sufficientlysecure.keychain.operations.ImportOperation;
|
|||||||
import org.sufficientlysecure.keychain.operations.results.ImportKeyResult;
|
import org.sufficientlysecure.keychain.operations.results.ImportKeyResult;
|
||||||
import org.sufficientlysecure.keychain.operations.results.OperationResult;
|
import org.sufficientlysecure.keychain.operations.results.OperationResult;
|
||||||
import org.sufficientlysecure.keychain.pgp.Progressable;
|
import org.sufficientlysecure.keychain.pgp.Progressable;
|
||||||
|
import org.sufficientlysecure.keychain.provider.KeyMetadataDao;
|
||||||
import org.sufficientlysecure.keychain.provider.KeyWritableRepository;
|
import org.sufficientlysecure.keychain.provider.KeyWritableRepository;
|
||||||
import org.sufficientlysecure.keychain.provider.LastUpdateInteractor;
|
|
||||||
import org.sufficientlysecure.keychain.service.ImportKeyringParcel;
|
import org.sufficientlysecure.keychain.service.ImportKeyringParcel;
|
||||||
import org.sufficientlysecure.keychain.service.input.CryptoInputParcel;
|
import org.sufficientlysecure.keychain.service.input.CryptoInputParcel;
|
||||||
import org.sufficientlysecure.keychain.ui.OrbotRequiredDialogActivity;
|
import org.sufficientlysecure.keychain.ui.OrbotRequiredDialogActivity;
|
||||||
@@ -49,14 +49,14 @@ public class KeyserverSyncWorker extends Worker {
|
|||||||
Constants.DEBUG_KEYSERVER_SYNC ? 2 : (int) TimeUnit.MINUTES.toSeconds(10);
|
Constants.DEBUG_KEYSERVER_SYNC ? 2 : (int) TimeUnit.MINUTES.toSeconds(10);
|
||||||
|
|
||||||
private AtomicBoolean cancellationSignal = new AtomicBoolean(false);
|
private AtomicBoolean cancellationSignal = new AtomicBoolean(false);
|
||||||
private LastUpdateInteractor lastUpdateInteractor;
|
private KeyMetadataDao keyMetadataDao;
|
||||||
private KeyWritableRepository keyWritableRepository;
|
private KeyWritableRepository keyWritableRepository;
|
||||||
private Preferences preferences;
|
private Preferences preferences;
|
||||||
|
|
||||||
@NonNull
|
@NonNull
|
||||||
@Override
|
@Override
|
||||||
public WorkerResult doWork() {
|
public WorkerResult doWork() {
|
||||||
lastUpdateInteractor = LastUpdateInteractor.create(getApplicationContext());
|
keyMetadataDao = KeyMetadataDao.create(getApplicationContext());
|
||||||
keyWritableRepository = KeyWritableRepository.create(getApplicationContext());
|
keyWritableRepository = KeyWritableRepository.create(getApplicationContext());
|
||||||
preferences = Preferences.getPreferences(getApplicationContext());
|
preferences = Preferences.getPreferences(getApplicationContext());
|
||||||
|
|
||||||
@@ -73,7 +73,7 @@ public class KeyserverSyncWorker extends Worker {
|
|||||||
Progressable notificationProgressable) {
|
Progressable notificationProgressable) {
|
||||||
long staleKeyThreshold = System.currentTimeMillis() - (isForceUpdate ? 0 : KEY_STALE_THRESHOLD_MILLIS);
|
long staleKeyThreshold = System.currentTimeMillis() - (isForceUpdate ? 0 : KEY_STALE_THRESHOLD_MILLIS);
|
||||||
List<byte[]> staleKeyFingerprints =
|
List<byte[]> staleKeyFingerprints =
|
||||||
lastUpdateInteractor.getFingerprintsForKeysOlderThan(staleKeyThreshold, TimeUnit.MILLISECONDS);
|
keyMetadataDao.getFingerprintsForKeysOlderThan(staleKeyThreshold, TimeUnit.MILLISECONDS);
|
||||||
List<ParcelableKeyRing> staleKeyParcelableKeyRings = fingerprintListToParcelableKeyRings(staleKeyFingerprints);
|
List<ParcelableKeyRing> staleKeyParcelableKeyRings = fingerprintListToParcelableKeyRings(staleKeyFingerprints);
|
||||||
|
|
||||||
if (isStopped()) { // if we've already been cancelled
|
if (isStopped()) { // if we've already been cancelled
|
||||||
|
|||||||
@@ -0,0 +1,28 @@
|
|||||||
|
package org.sufficientlysecure.keychain.model;
|
||||||
|
|
||||||
|
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
|
import android.support.annotation.NonNull;
|
||||||
|
|
||||||
|
import com.squareup.sqldelight.ColumnAdapter;
|
||||||
|
|
||||||
|
|
||||||
|
public final class CustomColumnAdapters {
|
||||||
|
|
||||||
|
private CustomColumnAdapters() { }
|
||||||
|
|
||||||
|
static final ColumnAdapter<Date,Long> DATE_ADAPTER = new ColumnAdapter<Date,Long>() {
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public Date decode(Long databaseValue) {
|
||||||
|
// Both SQLite and OpenPGP prefer a second granularity for timestamps - so we'll translate here
|
||||||
|
return new Date(databaseValue * 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Long encode(@NonNull Date value) {
|
||||||
|
return value.getTime() / 1000;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
package org.sufficientlysecure.keychain.model;
|
||||||
|
|
||||||
|
|
||||||
|
import com.google.auto.value.AutoValue;
|
||||||
|
import org.sufficientlysecure.keychain.KeyMetadataModel;
|
||||||
|
|
||||||
|
|
||||||
|
@AutoValue
|
||||||
|
public abstract class KeyMetadata implements KeyMetadataModel {
|
||||||
|
public static final Factory<KeyMetadata> FACTORY = new Factory<>(
|
||||||
|
AutoValue_KeyMetadata::new, CustomColumnAdapters.DATE_ADAPTER);
|
||||||
|
|
||||||
|
public boolean hasBeenUpdated() {
|
||||||
|
return last_updated() != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isPublished() {
|
||||||
|
if (last_updated() == null) {
|
||||||
|
throw new IllegalStateException("Cannot get publication state if key has never been updated!");
|
||||||
|
}
|
||||||
|
Boolean seenOnKeyservers = seen_on_keyservers();
|
||||||
|
return seenOnKeyservers != null && seenOnKeyservers;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -41,7 +41,7 @@ import org.sufficientlysecure.keychain.pgp.exception.PgpGeneralException;
|
|||||||
import org.sufficientlysecure.keychain.provider.CachedPublicKeyRing;
|
import org.sufficientlysecure.keychain.provider.CachedPublicKeyRing;
|
||||||
import org.sufficientlysecure.keychain.provider.KeyRepository.NotFoundException;
|
import org.sufficientlysecure.keychain.provider.KeyRepository.NotFoundException;
|
||||||
import org.sufficientlysecure.keychain.provider.KeyWritableRepository;
|
import org.sufficientlysecure.keychain.provider.KeyWritableRepository;
|
||||||
import org.sufficientlysecure.keychain.provider.LastUpdateInteractor;
|
import org.sufficientlysecure.keychain.provider.KeyMetadataDao;
|
||||||
import org.sufficientlysecure.keychain.service.CertifyActionsParcel;
|
import org.sufficientlysecure.keychain.service.CertifyActionsParcel;
|
||||||
import org.sufficientlysecure.keychain.service.CertifyActionsParcel.CertifyAction;
|
import org.sufficientlysecure.keychain.service.CertifyActionsParcel.CertifyAction;
|
||||||
import org.sufficientlysecure.keychain.service.ContactSyncAdapterService;
|
import org.sufficientlysecure.keychain.service.ContactSyncAdapterService;
|
||||||
@@ -62,13 +62,13 @@ import org.sufficientlysecure.keychain.util.Passphrase;
|
|||||||
* @see CertifyActionsParcel
|
* @see CertifyActionsParcel
|
||||||
*/
|
*/
|
||||||
public class CertifyOperation extends BaseReadWriteOperation<CertifyActionsParcel> {
|
public class CertifyOperation extends BaseReadWriteOperation<CertifyActionsParcel> {
|
||||||
private final LastUpdateInteractor lastUpdateInteractor;
|
private final KeyMetadataDao keyMetadataDao;
|
||||||
|
|
||||||
public CertifyOperation(Context context, KeyWritableRepository databaseInteractor, Progressable progressable, AtomicBoolean
|
public CertifyOperation(Context context, KeyWritableRepository keyWritableRepository, Progressable progressable, AtomicBoolean
|
||||||
cancelled) {
|
cancelled) {
|
||||||
super(context, databaseInteractor, progressable, cancelled);
|
super(context, keyWritableRepository, progressable, cancelled);
|
||||||
|
|
||||||
this.lastUpdateInteractor = LastUpdateInteractor.create(context);
|
this.keyMetadataDao = KeyMetadataDao.create(context);
|
||||||
}
|
}
|
||||||
|
|
||||||
@NonNull
|
@NonNull
|
||||||
@@ -234,7 +234,7 @@ public class CertifyOperation extends BaseReadWriteOperation<CertifyActionsParce
|
|||||||
log.add(uploadResult, 2);
|
log.add(uploadResult, 2);
|
||||||
|
|
||||||
if (uploadResult.success()) {
|
if (uploadResult.success()) {
|
||||||
lastUpdateInteractor.renewKeyLastUpdatedTime(certifiedKey.getMasterKeyId(), true);
|
keyMetadataDao.renewKeyLastUpdatedTime(certifiedKey.getMasterKeyId(), true);
|
||||||
|
|
||||||
uploadOk += 1;
|
uploadOk += 1;
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ import org.sufficientlysecure.keychain.pgp.Progressable;
|
|||||||
import org.sufficientlysecure.keychain.pgp.UncachedKeyRing;
|
import org.sufficientlysecure.keychain.pgp.UncachedKeyRing;
|
||||||
import org.sufficientlysecure.keychain.provider.KeyRepository.NotFoundException;
|
import org.sufficientlysecure.keychain.provider.KeyRepository.NotFoundException;
|
||||||
import org.sufficientlysecure.keychain.provider.KeyWritableRepository;
|
import org.sufficientlysecure.keychain.provider.KeyWritableRepository;
|
||||||
import org.sufficientlysecure.keychain.provider.LastUpdateInteractor;
|
import org.sufficientlysecure.keychain.provider.KeyMetadataDao;
|
||||||
import org.sufficientlysecure.keychain.service.ContactSyncAdapterService;
|
import org.sufficientlysecure.keychain.service.ContactSyncAdapterService;
|
||||||
import org.sufficientlysecure.keychain.service.SaveKeyringParcel;
|
import org.sufficientlysecure.keychain.service.SaveKeyringParcel;
|
||||||
import org.sufficientlysecure.keychain.service.UploadKeyringParcel;
|
import org.sufficientlysecure.keychain.service.UploadKeyringParcel;
|
||||||
@@ -57,14 +57,14 @@ import org.sufficientlysecure.keychain.util.ProgressScaler;
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public class EditKeyOperation extends BaseReadWriteOperation<SaveKeyringParcel> {
|
public class EditKeyOperation extends BaseReadWriteOperation<SaveKeyringParcel> {
|
||||||
private final LastUpdateInteractor lastUpdateInteractor;
|
private final KeyMetadataDao keyMetadataDao;
|
||||||
|
|
||||||
|
|
||||||
public EditKeyOperation(Context context, KeyWritableRepository databaseInteractor,
|
public EditKeyOperation(Context context, KeyWritableRepository databaseInteractor,
|
||||||
Progressable progressable, AtomicBoolean cancelled) {
|
Progressable progressable, AtomicBoolean cancelled) {
|
||||||
super(context, databaseInteractor, progressable, cancelled);
|
super(context, databaseInteractor, progressable, cancelled);
|
||||||
|
|
||||||
this.lastUpdateInteractor = LastUpdateInteractor.create(context);
|
this.keyMetadataDao = KeyMetadataDao.create(context);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -172,7 +172,7 @@ public class EditKeyOperation extends BaseReadWriteOperation<SaveKeyringParcel>
|
|||||||
log.add(saveResult, 1);
|
log.add(saveResult, 1);
|
||||||
|
|
||||||
if (isNewKey) {
|
if (isNewKey) {
|
||||||
lastUpdateInteractor.renewKeyLastUpdatedTime(ring.getMasterKeyId(), saveParcel.isShouldUpload());
|
keyMetadataDao.renewKeyLastUpdatedTime(ring.getMasterKeyId(), saveParcel.isShouldUpload());
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the save operation didn't succeed, exit here
|
// If the save operation didn't succeed, exit here
|
||||||
|
|||||||
@@ -55,7 +55,7 @@ import org.sufficientlysecure.keychain.pgp.Progressable;
|
|||||||
import org.sufficientlysecure.keychain.pgp.UncachedKeyRing;
|
import org.sufficientlysecure.keychain.pgp.UncachedKeyRing;
|
||||||
import org.sufficientlysecure.keychain.pgp.exception.PgpGeneralException;
|
import org.sufficientlysecure.keychain.pgp.exception.PgpGeneralException;
|
||||||
import org.sufficientlysecure.keychain.provider.KeyWritableRepository;
|
import org.sufficientlysecure.keychain.provider.KeyWritableRepository;
|
||||||
import org.sufficientlysecure.keychain.provider.LastUpdateInteractor;
|
import org.sufficientlysecure.keychain.provider.KeyMetadataDao;
|
||||||
import org.sufficientlysecure.keychain.service.ContactSyncAdapterService;
|
import org.sufficientlysecure.keychain.service.ContactSyncAdapterService;
|
||||||
import org.sufficientlysecure.keychain.service.ImportKeyringParcel;
|
import org.sufficientlysecure.keychain.service.ImportKeyringParcel;
|
||||||
import org.sufficientlysecure.keychain.service.input.CryptoInputParcel;
|
import org.sufficientlysecure.keychain.service.input.CryptoInputParcel;
|
||||||
@@ -90,7 +90,7 @@ public class ImportOperation extends BaseReadWriteOperation<ImportKeyringParcel>
|
|||||||
|
|
||||||
public static final String CACHE_FILE_NAME = "key_import.pcl";
|
public static final String CACHE_FILE_NAME = "key_import.pcl";
|
||||||
|
|
||||||
private final LastUpdateInteractor lastUpdateInteractor;
|
private final KeyMetadataDao keyMetadataDao;
|
||||||
|
|
||||||
private FacebookKeyserverClient facebookServer;
|
private FacebookKeyserverClient facebookServer;
|
||||||
private KeybaseKeyserverClient keybaseServer;
|
private KeybaseKeyserverClient keybaseServer;
|
||||||
@@ -98,14 +98,14 @@ public class ImportOperation extends BaseReadWriteOperation<ImportKeyringParcel>
|
|||||||
public ImportOperation(Context context, KeyWritableRepository databaseInteractor, Progressable progressable) {
|
public ImportOperation(Context context, KeyWritableRepository databaseInteractor, Progressable progressable) {
|
||||||
super(context, databaseInteractor, progressable);
|
super(context, databaseInteractor, progressable);
|
||||||
|
|
||||||
this.lastUpdateInteractor = LastUpdateInteractor.create(context);
|
this.keyMetadataDao = KeyMetadataDao.create(context);
|
||||||
}
|
}
|
||||||
|
|
||||||
public ImportOperation(Context context, KeyWritableRepository databaseInteractor,
|
public ImportOperation(Context context, KeyWritableRepository databaseInteractor,
|
||||||
Progressable progressable, AtomicBoolean cancelled) {
|
Progressable progressable, AtomicBoolean cancelled) {
|
||||||
super(context, databaseInteractor, progressable, cancelled);
|
super(context, databaseInteractor, progressable, cancelled);
|
||||||
|
|
||||||
this.lastUpdateInteractor = LastUpdateInteractor.create(context);
|
this.keyMetadataDao = KeyMetadataDao.create(context);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Overloaded functions for using progressable supplied in constructor during import
|
// Overloaded functions for using progressable supplied in constructor during import
|
||||||
@@ -200,7 +200,7 @@ public class ImportOperation extends BaseReadWriteOperation<ImportKeyringParcel>
|
|||||||
|
|
||||||
byte[] fingerprintHex = entry.getExpectedFingerprint();
|
byte[] fingerprintHex = entry.getExpectedFingerprint();
|
||||||
if (fingerprintHex != null) {
|
if (fingerprintHex != null) {
|
||||||
lastUpdateInteractor.renewKeyLastUpdatedTime(
|
keyMetadataDao.renewKeyLastUpdatedTime(
|
||||||
KeyFormattingUtils.getKeyIdFromFingerprint(fingerprintHex), false);
|
KeyFormattingUtils.getKeyIdFromFingerprint(fingerprintHex), false);
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
@@ -250,7 +250,7 @@ public class ImportOperation extends BaseReadWriteOperation<ImportKeyringParcel>
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!skipSave) {
|
if (!skipSave) {
|
||||||
lastUpdateInteractor.renewKeyLastUpdatedTime(key.getMasterKeyId(), keyWasDownloaded);
|
keyMetadataDao.renewKeyLastUpdatedTime(key.getMasterKeyId(), keyWasDownloaded);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -37,7 +37,6 @@ import org.sufficientlysecure.keychain.model.ApiApp;
|
|||||||
|
|
||||||
|
|
||||||
public class ApiDataAccessObject {
|
public class ApiDataAccessObject {
|
||||||
|
|
||||||
private final SupportSQLiteDatabase db;
|
private final SupportSQLiteDatabase db;
|
||||||
|
|
||||||
public ApiDataAccessObject(Context context) {
|
public ApiDataAccessObject(Context context) {
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ public class DatabaseNotifyManager {
|
|||||||
contentResolver.notifyChange(uri, null);
|
contentResolver.notifyChange(uri, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void notifyKeyserverStatusChange(long masterKeyId) {
|
public void notifyKeyMetadataChange(long masterKeyId) {
|
||||||
Uri uri = KeyRings.buildGenericKeyRingUri(masterKeyId);
|
Uri uri = KeyRings.buildGenericKeyRingUri(masterKeyId);
|
||||||
contentResolver.notifyChange(uri, null);
|
contentResolver.notifyChange(uri, null);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,69 @@
|
|||||||
|
package org.sufficientlysecure.keychain.provider;
|
||||||
|
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import android.arch.persistence.db.SupportSQLiteDatabase;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.database.Cursor;
|
||||||
|
|
||||||
|
import com.squareup.sqldelight.SqlDelightQuery;
|
||||||
|
import org.sufficientlysecure.keychain.KeyMetadataModel.ReplaceKeyMetadata;
|
||||||
|
import org.sufficientlysecure.keychain.model.KeyMetadata;
|
||||||
|
|
||||||
|
|
||||||
|
public class KeyMetadataDao {
|
||||||
|
private final SupportSQLiteDatabase db;
|
||||||
|
private DatabaseNotifyManager databaseNotifyManager;
|
||||||
|
|
||||||
|
|
||||||
|
public static KeyMetadataDao create(Context context) {
|
||||||
|
SupportSQLiteDatabase supportSQLiteDatabase = new KeychainDatabase(context).getWritableDatabase();
|
||||||
|
DatabaseNotifyManager databaseNotifyManager = DatabaseNotifyManager.create(context);
|
||||||
|
|
||||||
|
return new KeyMetadataDao(supportSQLiteDatabase, databaseNotifyManager);
|
||||||
|
}
|
||||||
|
|
||||||
|
private KeyMetadataDao(SupportSQLiteDatabase supportSQLiteDatabase, DatabaseNotifyManager databaseNotifyManager) {
|
||||||
|
this.db = supportSQLiteDatabase;
|
||||||
|
this.databaseNotifyManager = databaseNotifyManager;
|
||||||
|
}
|
||||||
|
|
||||||
|
public KeyMetadata getKeyMetadata(long masterKeyId) {
|
||||||
|
SqlDelightQuery query = KeyMetadata.FACTORY.selectByMasterKeyId(masterKeyId);
|
||||||
|
try (Cursor cursor = db.query(query)) {
|
||||||
|
if (cursor.moveToFirst()) {
|
||||||
|
return KeyMetadata.FACTORY.selectByMasterKeyIdMapper().map(cursor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void resetAllLastUpdatedTimes() {
|
||||||
|
new KeyMetadata.DeleteAllLastUpdatedTimes(db).execute();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void renewKeyLastUpdatedTime(long masterKeyId, boolean seenOnKeyservers) {
|
||||||
|
ReplaceKeyMetadata replaceStatement = new ReplaceKeyMetadata(db, KeyMetadata.FACTORY);
|
||||||
|
replaceStatement.bind(masterKeyId, new Date(), seenOnKeyservers);
|
||||||
|
replaceStatement.executeInsert();
|
||||||
|
|
||||||
|
databaseNotifyManager.notifyKeyMetadataChange(masterKeyId);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<byte[]> getFingerprintsForKeysOlderThan(long olderThan, TimeUnit timeUnit) {
|
||||||
|
SqlDelightQuery query = KeyMetadata.FACTORY.selectFingerprintsForKeysOlderThan(new Date(timeUnit.toMillis(olderThan)));
|
||||||
|
|
||||||
|
List<byte[]> fingerprintList = new ArrayList<>();
|
||||||
|
try (Cursor cursor = db.query(query)) {
|
||||||
|
while (cursor.moveToNext()) {
|
||||||
|
byte[] fingerprint = KeyMetadata.FACTORY.selectFingerprintsForKeysOlderThanMapper().map(cursor);
|
||||||
|
fingerprintList.add(fingerprint);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return fingerprintList;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -27,7 +27,6 @@ import android.content.ContentResolver;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.database.Cursor;
|
import android.database.Cursor;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.support.annotation.Nullable;
|
|
||||||
|
|
||||||
import org.bouncycastle.bcpg.ArmoredOutputStream;
|
import org.bouncycastle.bcpg.ArmoredOutputStream;
|
||||||
import org.sufficientlysecure.keychain.operations.results.OperationResult.LogType;
|
import org.sufficientlysecure.keychain.operations.results.OperationResult.LogType;
|
||||||
@@ -39,9 +38,7 @@ import org.sufficientlysecure.keychain.pgp.exception.PgpKeyNotFoundException;
|
|||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.Certs;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.Certs;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRingData;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRingData;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UpdatedKeys;
|
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainDatabase.Tables;
|
|
||||||
import timber.log.Timber;
|
import timber.log.Timber;
|
||||||
|
|
||||||
|
|
||||||
@@ -284,31 +281,6 @@ public class KeyRepository {
|
|||||||
return contentResolver;
|
return contentResolver;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
|
||||||
Long getLastUpdateTime(long masterKeyId) {
|
|
||||||
Cursor lastUpdatedCursor = contentResolver.query(
|
|
||||||
UpdatedKeys.CONTENT_URI,
|
|
||||||
new String[] { UpdatedKeys.LAST_UPDATED },
|
|
||||||
Tables.UPDATED_KEYS + "." + UpdatedKeys.MASTER_KEY_ID + " = ?",
|
|
||||||
new String[] { "" + masterKeyId },
|
|
||||||
null
|
|
||||||
);
|
|
||||||
if (lastUpdatedCursor == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
Long lastUpdateTime;
|
|
||||||
try {
|
|
||||||
if (!lastUpdatedCursor.moveToNext()) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
lastUpdateTime = lastUpdatedCursor.getLong(0);
|
|
||||||
} finally {
|
|
||||||
lastUpdatedCursor.close();
|
|
||||||
}
|
|
||||||
return lastUpdateTime;
|
|
||||||
}
|
|
||||||
|
|
||||||
public final byte[] loadPublicKeyRingData(long masterKeyId) throws NotFoundException {
|
public final byte[] loadPublicKeyRingData(long masterKeyId) throws NotFoundException {
|
||||||
byte[] data = (byte[]) getGenericDataOrNull(KeyRingData.buildPublicKeyRingUri(masterKeyId),
|
byte[] data = (byte[]) getGenericDataOrNull(KeyRingData.buildPublicKeyRingUri(masterKeyId),
|
||||||
KeyRingData.KEY_RING_DATA, FIELD_TYPE_BLOB);
|
KeyRingData.KEY_RING_DATA, FIELD_TYPE_BLOB);
|
||||||
|
|||||||
@@ -61,7 +61,6 @@ import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRingData;
|
|||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.KeySignatures;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.KeySignatures;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.Keys;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.Keys;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UpdatedKeys;
|
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
|
||||||
import org.sufficientlysecure.keychain.ui.util.KeyFormattingUtils;
|
import org.sufficientlysecure.keychain.ui.util.KeyFormattingUtils;
|
||||||
import org.sufficientlysecure.keychain.util.IterableIterator;
|
import org.sufficientlysecure.keychain.util.IterableIterator;
|
||||||
@@ -84,36 +83,30 @@ public class KeyWritableRepository extends KeyRepository {
|
|||||||
private static final int MAX_CACHED_KEY_SIZE = 1024 * 50;
|
private static final int MAX_CACHED_KEY_SIZE = 1024 * 50;
|
||||||
|
|
||||||
private final Context context;
|
private final Context context;
|
||||||
private final LastUpdateInteractor lastUpdateInteractor;
|
|
||||||
private final DatabaseNotifyManager databaseNotifyManager;
|
private final DatabaseNotifyManager databaseNotifyManager;
|
||||||
|
|
||||||
public static KeyWritableRepository create(Context context) {
|
public static KeyWritableRepository create(Context context) {
|
||||||
LocalPublicKeyStorage localPublicKeyStorage = LocalPublicKeyStorage.getInstance(context);
|
LocalPublicKeyStorage localPublicKeyStorage = LocalPublicKeyStorage.getInstance(context);
|
||||||
LocalSecretKeyStorage localSecretKeyStorage = LocalSecretKeyStorage.getInstance(context);
|
LocalSecretKeyStorage localSecretKeyStorage = LocalSecretKeyStorage.getInstance(context);
|
||||||
DatabaseNotifyManager databaseNotifyManager = DatabaseNotifyManager.create(context);
|
DatabaseNotifyManager databaseNotifyManager = DatabaseNotifyManager.create(context);
|
||||||
LastUpdateInteractor lastUpdateInteractor = LastUpdateInteractor.create(context);
|
return new KeyWritableRepository(context, localPublicKeyStorage, localSecretKeyStorage, databaseNotifyManager);
|
||||||
return new KeyWritableRepository(context, localPublicKeyStorage, localSecretKeyStorage, databaseNotifyManager,
|
|
||||||
lastUpdateInteractor);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
KeyWritableRepository(Context context,
|
KeyWritableRepository(Context context,
|
||||||
LocalPublicKeyStorage localPublicKeyStorage,
|
LocalPublicKeyStorage localPublicKeyStorage,
|
||||||
LocalSecretKeyStorage localSecretKeyStorage,
|
LocalSecretKeyStorage localSecretKeyStorage,
|
||||||
DatabaseNotifyManager databaseNotifyManager,
|
DatabaseNotifyManager databaseNotifyManager) {
|
||||||
LastUpdateInteractor lastUpdateInteractor) {
|
this(context, localPublicKeyStorage, localSecretKeyStorage, databaseNotifyManager, new OperationLog(), 0);
|
||||||
this(context, localPublicKeyStorage, localSecretKeyStorage, databaseNotifyManager, lastUpdateInteractor,
|
|
||||||
new OperationLog(), 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private KeyWritableRepository(Context context, LocalPublicKeyStorage localPublicKeyStorage,
|
private KeyWritableRepository(Context context, LocalPublicKeyStorage localPublicKeyStorage,
|
||||||
LocalSecretKeyStorage localSecretKeyStorage, DatabaseNotifyManager databaseNotifyManager,
|
LocalSecretKeyStorage localSecretKeyStorage, DatabaseNotifyManager databaseNotifyManager,
|
||||||
LastUpdateInteractor lastUpdateInteractor, OperationLog log, int indent) {
|
OperationLog log, int indent) {
|
||||||
super(context.getContentResolver(), localPublicKeyStorage, localSecretKeyStorage, log, indent);
|
super(context.getContentResolver(), localPublicKeyStorage, localSecretKeyStorage, log, indent);
|
||||||
|
|
||||||
this.context = context;
|
this.context = context;
|
||||||
this.databaseNotifyManager = databaseNotifyManager;
|
this.databaseNotifyManager = databaseNotifyManager;
|
||||||
this.lastUpdateInteractor = lastUpdateInteractor;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private LongSparseArray<CanonicalizedPublicKey> getTrustedMasterKeys() {
|
private LongSparseArray<CanonicalizedPublicKey> getTrustedMasterKeys() {
|
||||||
@@ -532,11 +525,6 @@ public class KeyWritableRepository extends KeyRepository {
|
|||||||
mIndent -= 1;
|
mIndent -= 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
ContentProviderOperation lastUpdateReinsertOp = getLastUpdatedReinsertOperationByMasterKeyId(masterKeyId);
|
|
||||||
if (lastUpdateReinsertOp != null) {
|
|
||||||
operations.add(lastUpdateReinsertOp);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// delete old version of this keyRing (from database only!), which also deletes all keys and userIds on cascade
|
// delete old version of this keyRing (from database only!), which also deletes all keys and userIds on cascade
|
||||||
int deleted = contentResolver.delete(
|
int deleted = contentResolver.delete(
|
||||||
@@ -567,26 +555,6 @@ public class KeyWritableRepository extends KeyRepository {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private ContentProviderOperation getLastUpdatedReinsertOperationByMasterKeyId(long masterKeyId) {
|
|
||||||
Long lastUpdateTime = getLastUpdateTime(masterKeyId);
|
|
||||||
if (lastUpdateTime == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
Boolean seenOnKeyservers = lastUpdateInteractor.getSeenOnKeyservers(masterKeyId);
|
|
||||||
|
|
||||||
ContentValues lastUpdatedEntry = new ContentValues(2);
|
|
||||||
lastUpdatedEntry.put(UpdatedKeys.MASTER_KEY_ID, masterKeyId);
|
|
||||||
lastUpdatedEntry.put(UpdatedKeys.LAST_UPDATED, lastUpdateTime);
|
|
||||||
if (seenOnKeyservers != null){
|
|
||||||
lastUpdatedEntry.put(UpdatedKeys.SEEN_ON_KEYSERVERS, seenOnKeyservers);
|
|
||||||
}
|
|
||||||
return ContentProviderOperation
|
|
||||||
.newInsert(UpdatedKeys.CONTENT_URI)
|
|
||||||
.withValues(lastUpdatedEntry)
|
|
||||||
.build();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void writePublicKeyRing(CanonicalizedPublicKeyRing keyRing, long masterKeyId,
|
private void writePublicKeyRing(CanonicalizedPublicKeyRing keyRing, long masterKeyId,
|
||||||
ArrayList<ContentProviderOperation> operations) throws IOException {
|
ArrayList<ContentProviderOperation> operations) throws IOException {
|
||||||
byte[] encodedKey = keyRing.getEncoded();
|
byte[] encodedKey = keyRing.getEncoded();
|
||||||
|
|||||||
@@ -263,16 +263,6 @@ public class KeychainContract {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class UpdatedKeys implements UpdatedKeysColumns, BaseColumns {
|
|
||||||
public static final Uri CONTENT_URI = BASE_CONTENT_URI_INTERNAL.buildUpon()
|
|
||||||
.appendPath(BASE_UPDATED_KEYS).build();
|
|
||||||
|
|
||||||
public static final String CONTENT_TYPE
|
|
||||||
= "vnd.android.cursor.dir/vnd.org.sufficientlysecure.keychain.provider.updated_keys";
|
|
||||||
public static final String CONTENT_ITEM_TYPE
|
|
||||||
= "vnd.android.cursor.item/vnd.org.sufficientlysecure.keychain.provider.updated_keys";
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class KeySignatures implements KeySignaturesColumns, BaseColumns {
|
public static class KeySignatures implements KeySignaturesColumns, BaseColumns {
|
||||||
public static final Uri CONTENT_URI = BASE_CONTENT_URI_INTERNAL.buildUpon()
|
public static final Uri CONTENT_URI = BASE_CONTENT_URI_INTERNAL.buildUpon()
|
||||||
.appendPath(BASE_KEY_SIGNATURES).build();
|
.appendPath(BASE_KEY_SIGNATURES).build();
|
||||||
|
|||||||
@@ -34,6 +34,9 @@ import android.database.sqlite.SQLiteException;
|
|||||||
import android.provider.BaseColumns;
|
import android.provider.BaseColumns;
|
||||||
|
|
||||||
import org.sufficientlysecure.keychain.Constants;
|
import org.sufficientlysecure.keychain.Constants;
|
||||||
|
import org.sufficientlysecure.keychain.KeyMetadataModel;
|
||||||
|
import org.sufficientlysecure.keychain.KeyRingsPublicModel;
|
||||||
|
import org.sufficientlysecure.keychain.model.ApiApp;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.ApiAppsAllowedKeysColumns;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.ApiAppsAllowedKeysColumns;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.ApiAutocryptPeerColumns;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.ApiAutocryptPeerColumns;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.CertsColumns;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.CertsColumns;
|
||||||
@@ -57,7 +60,7 @@ import timber.log.Timber;
|
|||||||
*/
|
*/
|
||||||
public class KeychainDatabase {
|
public class KeychainDatabase {
|
||||||
private static final String DATABASE_NAME = "openkeychain.db";
|
private static final String DATABASE_NAME = "openkeychain.db";
|
||||||
private static final int DATABASE_VERSION = 26;
|
private static final int DATABASE_VERSION = 27;
|
||||||
private final SupportSQLiteOpenHelper supportSQLiteOpenHelper;
|
private final SupportSQLiteOpenHelper supportSQLiteOpenHelper;
|
||||||
private Context context;
|
private Context context;
|
||||||
|
|
||||||
@@ -226,15 +229,16 @@ public class KeychainDatabase {
|
|||||||
private void onCreate(SupportSQLiteDatabase db) {
|
private void onCreate(SupportSQLiteDatabase db) {
|
||||||
Timber.w("Creating database...");
|
Timber.w("Creating database...");
|
||||||
|
|
||||||
db.execSQL(CREATE_KEYRINGS_PUBLIC);
|
db.execSQL(KeyRingsPublicModel.CREATE_TABLE);
|
||||||
db.execSQL(CREATE_KEYS);
|
db.execSQL(CREATE_KEYS);
|
||||||
db.execSQL(CREATE_USER_PACKETS);
|
db.execSQL(CREATE_USER_PACKETS);
|
||||||
db.execSQL(CREATE_CERTS);
|
db.execSQL(CREATE_CERTS);
|
||||||
db.execSQL(CREATE_UPDATE_KEYS);
|
db.execSQL(KeyMetadataModel.CREATE_TABLE);
|
||||||
db.execSQL(CREATE_KEY_SIGNATURES);
|
db.execSQL(CREATE_KEY_SIGNATURES);
|
||||||
db.execSQL(CREATE_API_APPS_ALLOWED_KEYS);
|
db.execSQL(CREATE_API_APPS_ALLOWED_KEYS);
|
||||||
db.execSQL(CREATE_OVERRIDDEN_WARNINGS);
|
db.execSQL(CREATE_OVERRIDDEN_WARNINGS);
|
||||||
db.execSQL(CREATE_API_AUTOCRYPT_PEERS);
|
db.execSQL(CREATE_API_AUTOCRYPT_PEERS);
|
||||||
|
db.execSQL(ApiApp.CREATE_TABLE);
|
||||||
|
|
||||||
db.execSQL("CREATE INDEX keys_by_rank ON keys (" + KeysColumns.RANK + ", " + KeysColumns.MASTER_KEY_ID + ");");
|
db.execSQL("CREATE INDEX keys_by_rank ON keys (" + KeysColumns.RANK + ", " + KeysColumns.MASTER_KEY_ID + ");");
|
||||||
db.execSQL("CREATE INDEX uids_by_rank ON user_packets (" + UserPacketsColumns.RANK + ", "
|
db.execSQL("CREATE INDEX uids_by_rank ON user_packets (" + UserPacketsColumns.RANK + ", "
|
||||||
@@ -459,6 +463,9 @@ public class KeychainDatabase {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case 26: {
|
||||||
|
migrateUpdatedKeysToKeyMetadataTable(db);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -476,6 +483,11 @@ public class KeychainDatabase {
|
|||||||
// db.execSQL("DROP TABLE keyrings_secret");
|
// db.execSQL("DROP TABLE keyrings_secret");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void migrateUpdatedKeysToKeyMetadataTable(SupportSQLiteDatabase db) {
|
||||||
|
db.execSQL("ALTER TABLE updated_keys RENAME TO key_metadata;");
|
||||||
|
db.execSQL("UPDATE key_metadata SET last_updated = last_updated * 1000;");
|
||||||
|
}
|
||||||
|
|
||||||
public void onDowngrade(SupportSQLiteDatabase db, int oldVersion, int newVersion) {
|
public void onDowngrade(SupportSQLiteDatabase db, int oldVersion, int newVersion) {
|
||||||
// Downgrade is ok for the debug version, makes it easier to work with branches
|
// Downgrade is ok for the debug version, makes it easier to work with branches
|
||||||
if (Constants.DEBUG) {
|
if (Constants.DEBUG) {
|
||||||
|
|||||||
@@ -45,7 +45,6 @@ import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRingData;
|
|||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.KeySignatures;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.KeySignatures;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.Keys;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.Keys;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UpdatedKeys;
|
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPacketsColumns;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPacketsColumns;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainDatabase.Tables;
|
import org.sufficientlysecure.keychain.provider.KeychainDatabase.Tables;
|
||||||
@@ -75,9 +74,6 @@ public class KeychainProvider extends ContentProvider implements SimpleContentRe
|
|||||||
private static final int KEY_RINGS_FIND_BY_USER_ID = 402;
|
private static final int KEY_RINGS_FIND_BY_USER_ID = 402;
|
||||||
private static final int KEY_RINGS_FILTER_BY_SIGNER = 403;
|
private static final int KEY_RINGS_FILTER_BY_SIGNER = 403;
|
||||||
|
|
||||||
private static final int UPDATED_KEYS = 500;
|
|
||||||
private static final int UPDATED_KEYS_SPECIFIC = 501;
|
|
||||||
|
|
||||||
private static final int AUTOCRYPT_PEERS_BY_MASTER_KEY_ID = 601;
|
private static final int AUTOCRYPT_PEERS_BY_MASTER_KEY_ID = 601;
|
||||||
private static final int AUTOCRYPT_PEERS_BY_PACKAGE_NAME = 602;
|
private static final int AUTOCRYPT_PEERS_BY_PACKAGE_NAME = 602;
|
||||||
private static final int AUTOCRYPT_PEERS_BY_PACKAGE_NAME_AND_TRUST_ID = 603;
|
private static final int AUTOCRYPT_PEERS_BY_PACKAGE_NAME_AND_TRUST_ID = 603;
|
||||||
@@ -192,14 +188,6 @@ public class KeychainProvider extends ContentProvider implements SimpleContentRe
|
|||||||
matcher.addURI(authority, KeychainContract.BASE_AUTOCRYPT_PEERS + "/" +
|
matcher.addURI(authority, KeychainContract.BASE_AUTOCRYPT_PEERS + "/" +
|
||||||
KeychainContract.PATH_BY_PACKAGE_NAME + "/*/*", AUTOCRYPT_PEERS_BY_PACKAGE_NAME_AND_TRUST_ID);
|
KeychainContract.PATH_BY_PACKAGE_NAME + "/*/*", AUTOCRYPT_PEERS_BY_PACKAGE_NAME_AND_TRUST_ID);
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* to access table containing last updated dates of keys
|
|
||||||
*/
|
|
||||||
matcher.addURI(authority, KeychainContract.BASE_UPDATED_KEYS, UPDATED_KEYS);
|
|
||||||
matcher.addURI(authority, KeychainContract.BASE_UPDATED_KEYS + "/*", UPDATED_KEYS_SPECIFIC);
|
|
||||||
|
|
||||||
|
|
||||||
matcher.addURI(authority, KeychainContract.BASE_KEY_SIGNATURES, KEY_SIGNATURES);
|
matcher.addURI(authority, KeychainContract.BASE_KEY_SIGNATURES, KEY_SIGNATURES);
|
||||||
|
|
||||||
|
|
||||||
@@ -239,12 +227,6 @@ public class KeychainProvider extends ContentProvider implements SimpleContentRe
|
|||||||
case KEY_RING_USER_IDS:
|
case KEY_RING_USER_IDS:
|
||||||
return UserPackets.CONTENT_TYPE;
|
return UserPackets.CONTENT_TYPE;
|
||||||
|
|
||||||
case UPDATED_KEYS:
|
|
||||||
return UpdatedKeys.CONTENT_TYPE;
|
|
||||||
|
|
||||||
case UPDATED_KEYS_SPECIFIC:
|
|
||||||
return UpdatedKeys.CONTENT_ITEM_TYPE;
|
|
||||||
|
|
||||||
case KEY_SIGNATURES:
|
case KEY_SIGNATURES:
|
||||||
return KeySignatures.CONTENT_TYPE;
|
return KeySignatures.CONTENT_TYPE;
|
||||||
|
|
||||||
@@ -727,30 +709,6 @@ public class KeychainProvider extends ContentProvider implements SimpleContentRe
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case UPDATED_KEYS:
|
|
||||||
case UPDATED_KEYS_SPECIFIC: {
|
|
||||||
HashMap<String, String> projectionMap = new HashMap<>();
|
|
||||||
projectionMap.put(UpdatedKeys.MASTER_KEY_ID,
|
|
||||||
Tables.UPDATED_KEYS + "." + UpdatedKeys.MASTER_KEY_ID + " AS " + UpdatedKeys.MASTER_KEY_ID);
|
|
||||||
projectionMap.put(UpdatedKeys.LAST_UPDATED,
|
|
||||||
Tables.UPDATED_KEYS + "." + UpdatedKeys.LAST_UPDATED + " AS " + UpdatedKeys.LAST_UPDATED);
|
|
||||||
projectionMap.put(UpdatedKeys.SEEN_ON_KEYSERVERS,
|
|
||||||
Tables.UPDATED_KEYS + "." + UpdatedKeys.SEEN_ON_KEYSERVERS + " AS " + UpdatedKeys.SEEN_ON_KEYSERVERS);
|
|
||||||
projectionMap.put(UpdatedKeys.FINGERPRINT,
|
|
||||||
Tables.KEYS + "." + Keys.FINGERPRINT + " AS " + UpdatedKeys.FINGERPRINT);
|
|
||||||
qb.setProjectionMap(projectionMap);
|
|
||||||
|
|
||||||
qb.setTables(Tables.UPDATED_KEYS +
|
|
||||||
" LEFT JOIN " + Tables.KEYS +
|
|
||||||
" ON (" + Tables.KEYS + "." + Keys.KEY_ID + " = " + Tables.UPDATED_KEYS + "." + UpdatedKeys.MASTER_KEY_ID + ")"
|
|
||||||
);
|
|
||||||
|
|
||||||
if (match == UPDATED_KEYS_SPECIFIC) {
|
|
||||||
qb.appendWhere(Tables.UPDATED_KEYS + "." + UpdatedKeys.MASTER_KEY_ID + " = ");
|
|
||||||
qb.appendWhereEscapeString(uri.getPathSegments().get(1));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
default: {
|
default: {
|
||||||
throw new IllegalArgumentException("Unknown URI " + uri + " (" + match + ")");
|
throw new IllegalArgumentException("Unknown URI " + uri + " (" + match + ")");
|
||||||
}
|
}
|
||||||
@@ -835,17 +793,6 @@ public class KeychainProvider extends ContentProvider implements SimpleContentRe
|
|||||||
keyId = values.getAsLong(Certs.MASTER_KEY_ID);
|
keyId = values.getAsLong(Certs.MASTER_KEY_ID);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case UPDATED_KEYS: {
|
|
||||||
keyId = values.getAsLong(UpdatedKeys.MASTER_KEY_ID);
|
|
||||||
try {
|
|
||||||
db.insert(Tables.UPDATED_KEYS, SQLiteDatabase.CONFLICT_FAIL, values);
|
|
||||||
} catch (SQLiteConstraintException e) {
|
|
||||||
db.update(Tables.UPDATED_KEYS, SQLiteDatabase.CONFLICT_IGNORE, values,
|
|
||||||
UpdatedKeys.MASTER_KEY_ID + " = ?", new String[] { Long.toString(keyId) });
|
|
||||||
}
|
|
||||||
rowUri = UpdatedKeys.CONTENT_URI;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case KEY_SIGNATURES: {
|
case KEY_SIGNATURES: {
|
||||||
db.insert(Tables.KEY_SIGNATURES, SQLiteDatabase.CONFLICT_FAIL, values);
|
db.insert(Tables.KEY_SIGNATURES, SQLiteDatabase.CONFLICT_FAIL, values);
|
||||||
rowUri = KeySignatures.CONTENT_URI;
|
rowUri = KeySignatures.CONTENT_URI;
|
||||||
@@ -953,19 +900,6 @@ public class KeychainProvider extends ContentProvider implements SimpleContentRe
|
|||||||
count = db.update(Tables.KEYS, SQLiteDatabase.CONFLICT_FAIL, values, actualSelection, selectionArgs);
|
count = db.update(Tables.KEYS, SQLiteDatabase.CONFLICT_FAIL, values, actualSelection, selectionArgs);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case UPDATED_KEYS: {
|
|
||||||
if (values.size() != 2 ||
|
|
||||||
!values.containsKey(UpdatedKeys.SEEN_ON_KEYSERVERS) ||
|
|
||||||
!values.containsKey(UpdatedKeys.LAST_UPDATED) ||
|
|
||||||
values.get(UpdatedKeys.LAST_UPDATED) != null ||
|
|
||||||
values.get(UpdatedKeys.SEEN_ON_KEYSERVERS) != null ||
|
|
||||||
selection != null || selectionArgs != null) {
|
|
||||||
throw new UnsupportedOperationException("can only reset all keys");
|
|
||||||
}
|
|
||||||
|
|
||||||
db.update(Tables.UPDATED_KEYS, SQLiteDatabase.CONFLICT_FAIL, values, null, null);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case AUTOCRYPT_PEERS_BY_PACKAGE_NAME_AND_TRUST_ID: {
|
case AUTOCRYPT_PEERS_BY_PACKAGE_NAME_AND_TRUST_ID: {
|
||||||
String packageName = uri.getPathSegments().get(2);
|
String packageName = uri.getPathSegments().get(2);
|
||||||
String identifier = uri.getLastPathSegment();
|
String identifier = uri.getLastPathSegment();
|
||||||
|
|||||||
@@ -1,105 +0,0 @@
|
|||||||
package org.sufficientlysecure.keychain.provider;
|
|
||||||
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.GregorianCalendar;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import android.content.ContentResolver;
|
|
||||||
import android.content.ContentValues;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.database.Cursor;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.support.annotation.Nullable;
|
|
||||||
|
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UpdatedKeys;
|
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainDatabase.Tables;
|
|
||||||
|
|
||||||
|
|
||||||
public class LastUpdateInteractor {
|
|
||||||
private final ContentResolver contentResolver;
|
|
||||||
private final DatabaseNotifyManager databaseNotifyManager;
|
|
||||||
|
|
||||||
|
|
||||||
public static LastUpdateInteractor create(Context context) {
|
|
||||||
return new LastUpdateInteractor(context.getContentResolver(), DatabaseNotifyManager.create(context));
|
|
||||||
}
|
|
||||||
|
|
||||||
private LastUpdateInteractor(ContentResolver contentResolver, DatabaseNotifyManager databaseNotifyManager) {
|
|
||||||
this.contentResolver = contentResolver;
|
|
||||||
this.databaseNotifyManager = databaseNotifyManager;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public Boolean getSeenOnKeyservers(long masterKeyId) {
|
|
||||||
Cursor cursor = contentResolver.query(
|
|
||||||
UpdatedKeys.CONTENT_URI,
|
|
||||||
new String[] { UpdatedKeys.SEEN_ON_KEYSERVERS },
|
|
||||||
Tables.UPDATED_KEYS + "." + UpdatedKeys.MASTER_KEY_ID + " = ?",
|
|
||||||
new String[] { "" + masterKeyId },
|
|
||||||
null
|
|
||||||
);
|
|
||||||
if (cursor == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
Boolean seenOnKeyservers;
|
|
||||||
try {
|
|
||||||
if (!cursor.moveToNext()) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
seenOnKeyservers = cursor.isNull(0) ? null : cursor.getInt(0) != 0;
|
|
||||||
} finally {
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
return seenOnKeyservers;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void resetAllLastUpdatedTimes() {
|
|
||||||
ContentValues values = new ContentValues();
|
|
||||||
values.putNull(UpdatedKeys.LAST_UPDATED);
|
|
||||||
values.putNull(UpdatedKeys.SEEN_ON_KEYSERVERS);
|
|
||||||
contentResolver.update(UpdatedKeys.CONTENT_URI, values, null, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public Uri renewKeyLastUpdatedTime(long masterKeyId, boolean seenOnKeyservers) {
|
|
||||||
boolean isFirstKeyserverStatusCheck = getSeenOnKeyservers(masterKeyId) == null;
|
|
||||||
|
|
||||||
ContentValues values = new ContentValues();
|
|
||||||
values.put(UpdatedKeys.MASTER_KEY_ID, masterKeyId);
|
|
||||||
values.put(UpdatedKeys.LAST_UPDATED, GregorianCalendar.getInstance().getTimeInMillis() / 1000);
|
|
||||||
if (seenOnKeyservers || isFirstKeyserverStatusCheck) {
|
|
||||||
values.put(UpdatedKeys.SEEN_ON_KEYSERVERS, seenOnKeyservers);
|
|
||||||
}
|
|
||||||
|
|
||||||
// this will actually update/replace, doing the right thing™ for seenOnKeyservers value
|
|
||||||
// see `KeychainProvider.insert()`
|
|
||||||
Uri insert = contentResolver.insert(UpdatedKeys.CONTENT_URI, values);
|
|
||||||
databaseNotifyManager.notifyKeyserverStatusChange(masterKeyId);
|
|
||||||
return insert;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<byte[]> getFingerprintsForKeysOlderThan(long olderThan, TimeUnit timeUnit) {
|
|
||||||
Cursor outdatedKeysCursor = contentResolver.query(
|
|
||||||
KeychainContract.UpdatedKeys.CONTENT_URI,
|
|
||||||
new String[] { KeychainContract.UpdatedKeys.FINGERPRINT, },
|
|
||||||
KeychainContract.UpdatedKeys.LAST_UPDATED + " < ?",
|
|
||||||
new String[] { Long.toString(timeUnit.toSeconds(olderThan)) },
|
|
||||||
null
|
|
||||||
);
|
|
||||||
|
|
||||||
List<byte[]> fingerprintList = new ArrayList<>();
|
|
||||||
if (outdatedKeysCursor == null) {
|
|
||||||
return fingerprintList;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (outdatedKeysCursor.moveToNext()) {
|
|
||||||
byte[] fingerprint = outdatedKeysCursor.getBlob(0);
|
|
||||||
fingerprintList.add(fingerprint);
|
|
||||||
}
|
|
||||||
outdatedKeysCursor.close();
|
|
||||||
|
|
||||||
return fingerprintList;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -43,7 +43,7 @@ import android.widget.TextView;
|
|||||||
|
|
||||||
import org.sufficientlysecure.keychain.R;
|
import org.sufficientlysecure.keychain.R;
|
||||||
import org.sufficientlysecure.keychain.keyimport.HkpKeyserverAddress;
|
import org.sufficientlysecure.keychain.keyimport.HkpKeyserverAddress;
|
||||||
import org.sufficientlysecure.keychain.provider.LastUpdateInteractor;
|
import org.sufficientlysecure.keychain.provider.KeyMetadataDao;
|
||||||
import org.sufficientlysecure.keychain.ui.dialog.AddEditKeyserverDialogFragment;
|
import org.sufficientlysecure.keychain.ui.dialog.AddEditKeyserverDialogFragment;
|
||||||
import org.sufficientlysecure.keychain.ui.util.FormattingUtils;
|
import org.sufficientlysecure.keychain.ui.util.FormattingUtils;
|
||||||
import org.sufficientlysecure.keychain.ui.util.Notify;
|
import org.sufficientlysecure.keychain.ui.util.Notify;
|
||||||
@@ -63,7 +63,7 @@ public class SettingsKeyserverFragment extends Fragment implements RecyclerItemC
|
|||||||
private List<HkpKeyserverAddress> mKeyservers;
|
private List<HkpKeyserverAddress> mKeyservers;
|
||||||
private KeyserverListAdapter mAdapter;
|
private KeyserverListAdapter mAdapter;
|
||||||
|
|
||||||
private LastUpdateInteractor lastUpdateInteractor;
|
private KeyMetadataDao keyMetadataDao;
|
||||||
|
|
||||||
public static SettingsKeyserverFragment newInstance(ArrayList<HkpKeyserverAddress> keyservers) {
|
public static SettingsKeyserverFragment newInstance(ArrayList<HkpKeyserverAddress> keyservers) {
|
||||||
Bundle args = new Bundle();
|
Bundle args = new Bundle();
|
||||||
@@ -78,7 +78,7 @@ public class SettingsKeyserverFragment extends Fragment implements RecyclerItemC
|
|||||||
@Override
|
@Override
|
||||||
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle
|
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle
|
||||||
savedInstanceState) {
|
savedInstanceState) {
|
||||||
lastUpdateInteractor = LastUpdateInteractor.create(getContext());
|
keyMetadataDao = KeyMetadataDao.create(getContext());
|
||||||
|
|
||||||
return inflater.inflate(R.layout.settings_keyserver_fragment, null);
|
return inflater.inflate(R.layout.settings_keyserver_fragment, null);
|
||||||
}
|
}
|
||||||
@@ -230,7 +230,7 @@ public class SettingsKeyserverFragment extends Fragment implements RecyclerItemC
|
|||||||
Preferences.getPreferences(getActivity()).setKeyServers(mKeyserversMutable);
|
Preferences.getPreferences(getActivity()).setKeyServers(mKeyserversMutable);
|
||||||
mKeyservers = Collections.unmodifiableList(new ArrayList<>(mKeyserversMutable));
|
mKeyservers = Collections.unmodifiableList(new ArrayList<>(mKeyserversMutable));
|
||||||
|
|
||||||
lastUpdateInteractor.resetAllLastUpdatedTimes();
|
keyMetadataDao.resetAllLastUpdatedTimes();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|||||||
@@ -39,10 +39,10 @@ import android.view.ViewGroup;
|
|||||||
|
|
||||||
import org.sufficientlysecure.keychain.R;
|
import org.sufficientlysecure.keychain.R;
|
||||||
import org.sufficientlysecure.keychain.compatibility.DialogFragmentWorkaround;
|
import org.sufficientlysecure.keychain.compatibility.DialogFragmentWorkaround;
|
||||||
|
import org.sufficientlysecure.keychain.model.KeyMetadata;
|
||||||
import org.sufficientlysecure.keychain.operations.results.OperationResult;
|
import org.sufficientlysecure.keychain.operations.results.OperationResult;
|
||||||
import org.sufficientlysecure.keychain.ui.base.LoaderFragment;
|
import org.sufficientlysecure.keychain.ui.base.LoaderFragment;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.IdentityDao.IdentityInfo;
|
import org.sufficientlysecure.keychain.ui.keyview.loader.IdentityDao.IdentityInfo;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.KeyserverStatusDao.KeyserverStatus;
|
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.SubkeyStatusDao.KeySubkeyStatus;
|
import org.sufficientlysecure.keychain.ui.keyview.loader.SubkeyStatusDao.KeySubkeyStatus;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.SystemContactDao.SystemContactInfo;
|
import org.sufficientlysecure.keychain.ui.keyview.loader.SystemContactDao.SystemContactInfo;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.presenter.IdentitiesPresenter;
|
import org.sufficientlysecure.keychain.ui.keyview.presenter.IdentitiesPresenter;
|
||||||
@@ -108,7 +108,7 @@ public class ViewKeyFragment extends LoaderFragment implements ViewKeyMvpView, O
|
|||||||
private LiveData<List<IdentityInfo>> identityInfo;
|
private LiveData<List<IdentityInfo>> identityInfo;
|
||||||
private LiveData<KeySubkeyStatus> subkeyStatus;
|
private LiveData<KeySubkeyStatus> subkeyStatus;
|
||||||
private LiveData<SystemContactInfo> systemContactInfo;
|
private LiveData<SystemContactInfo> systemContactInfo;
|
||||||
private LiveData<KeyserverStatus> keyserverStatus;
|
private LiveData<KeyMetadata> keyserverStatus;
|
||||||
|
|
||||||
LiveData<List<IdentityInfo>> getIdentityInfo(IdentitiesPresenter identitiesPresenter) {
|
LiveData<List<IdentityInfo>> getIdentityInfo(IdentitiesPresenter identitiesPresenter) {
|
||||||
if (identityInfo == null) {
|
if (identityInfo == null) {
|
||||||
@@ -131,7 +131,7 @@ public class ViewKeyFragment extends LoaderFragment implements ViewKeyMvpView, O
|
|||||||
return systemContactInfo;
|
return systemContactInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
LiveData<KeyserverStatus> getKeyserverStatus(KeyserverStatusPresenter keyserverStatusPresenter) {
|
LiveData<KeyMetadata> getKeyserverStatus(KeyserverStatusPresenter keyserverStatusPresenter) {
|
||||||
if (keyserverStatus == null) {
|
if (keyserverStatus == null) {
|
||||||
keyserverStatus = keyserverStatusPresenter.getLiveDataInstance();
|
keyserverStatus = keyserverStatusPresenter.getLiveDataInstance();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,114 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (C) 2017 Schürmann & Breitmoser GbR
|
|
||||||
*
|
|
||||||
* 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.ui.keyview.loader;
|
|
||||||
|
|
||||||
|
|
||||||
import java.util.Date;
|
|
||||||
|
|
||||||
import android.content.ContentResolver;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.database.Cursor;
|
|
||||||
|
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.UpdatedKeys;
|
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainDatabase.Tables;
|
|
||||||
import timber.log.Timber;
|
|
||||||
|
|
||||||
|
|
||||||
public class KeyserverStatusDao {
|
|
||||||
public static final String[] PROJECTION = new String[] {
|
|
||||||
UpdatedKeys.LAST_UPDATED,
|
|
||||||
UpdatedKeys.SEEN_ON_KEYSERVERS
|
|
||||||
};
|
|
||||||
private static final int INDEX_LAST_UPDATED = 0;
|
|
||||||
private static final int INDEX_SEEN_ON_KEYSERVERS = 1;
|
|
||||||
|
|
||||||
|
|
||||||
private final ContentResolver contentResolver;
|
|
||||||
|
|
||||||
public static KeyserverStatusDao getInstance(Context context) {
|
|
||||||
ContentResolver contentResolver = context.getContentResolver();
|
|
||||||
return new KeyserverStatusDao(contentResolver);
|
|
||||||
}
|
|
||||||
|
|
||||||
private KeyserverStatusDao(ContentResolver contentResolver) {
|
|
||||||
this.contentResolver = contentResolver;
|
|
||||||
}
|
|
||||||
|
|
||||||
public KeyserverStatus getKeyserverStatus(long masterKeyId) {
|
|
||||||
Cursor cursor = contentResolver.query(UpdatedKeys.CONTENT_URI, PROJECTION,
|
|
||||||
Tables.UPDATED_KEYS + "." + UpdatedKeys.MASTER_KEY_ID + " = ?", new String[] { Long.toString(masterKeyId) }, null);
|
|
||||||
if (cursor == null) {
|
|
||||||
Timber.e("Error loading key items!");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (cursor.moveToFirst()) {
|
|
||||||
if (cursor.isNull(INDEX_SEEN_ON_KEYSERVERS) || cursor.isNull(INDEX_LAST_UPDATED)) {
|
|
||||||
return new KeyserverStatus(masterKeyId);
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean isPublished = cursor.getInt(INDEX_SEEN_ON_KEYSERVERS) != 0;
|
|
||||||
Date lastUpdated = new Date(cursor.getLong(INDEX_LAST_UPDATED) * 1000);
|
|
||||||
|
|
||||||
return new KeyserverStatus(masterKeyId, isPublished, lastUpdated);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new KeyserverStatus(masterKeyId);
|
|
||||||
} finally {
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class KeyserverStatus {
|
|
||||||
private final long masterKeyId;
|
|
||||||
private final boolean isPublished;
|
|
||||||
private final Date lastUpdated;
|
|
||||||
|
|
||||||
KeyserverStatus(long masterKeyId, boolean isPublished, Date lastUpdated) {
|
|
||||||
this.masterKeyId = masterKeyId;
|
|
||||||
this.isPublished = isPublished;
|
|
||||||
this.lastUpdated = lastUpdated;
|
|
||||||
}
|
|
||||||
|
|
||||||
KeyserverStatus(long masterKeyId) {
|
|
||||||
this.masterKeyId = masterKeyId;
|
|
||||||
this.isPublished = false;
|
|
||||||
this.lastUpdated = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
long getMasterKeyId() {
|
|
||||||
return masterKeyId;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean hasBeenUpdated() {
|
|
||||||
return lastUpdated != null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isPublished() {
|
|
||||||
if (lastUpdated == null) {
|
|
||||||
throw new IllegalStateException("Cannot get publication state if key has never been updated!");
|
|
||||||
}
|
|
||||||
return isPublished;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Date getLastUpdated() {
|
|
||||||
return lastUpdated;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -6,9 +6,10 @@ import java.util.List;
|
|||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
|
|
||||||
|
import org.sufficientlysecure.keychain.model.KeyMetadata;
|
||||||
|
import org.sufficientlysecure.keychain.provider.KeyMetadataDao;
|
||||||
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
|
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.IdentityDao.IdentityInfo;
|
import org.sufficientlysecure.keychain.ui.keyview.loader.IdentityDao.IdentityInfo;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.KeyserverStatusDao.KeyserverStatus;
|
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.SubkeyStatusDao.KeySubkeyStatus;
|
import org.sufficientlysecure.keychain.ui.keyview.loader.SubkeyStatusDao.KeySubkeyStatus;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.SubkeyStatusDao.SubKeyItem;
|
import org.sufficientlysecure.keychain.ui.keyview.loader.SubkeyStatusDao.SubKeyItem;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.SystemContactDao.SystemContactInfo;
|
import org.sufficientlysecure.keychain.ui.keyview.loader.SystemContactDao.SystemContactInfo;
|
||||||
@@ -78,21 +79,21 @@ public class ViewKeyLiveData {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class KeyserverStatusLiveData extends AsyncTaskLiveData<KeyserverStatus> {
|
public static class KeyserverStatusLiveData extends AsyncTaskLiveData<KeyMetadata> {
|
||||||
private final KeyserverStatusDao keyserverStatusDao;
|
private final KeyMetadataDao keyMetadataDao;
|
||||||
|
|
||||||
private final long masterKeyId;
|
private final long masterKeyId;
|
||||||
|
|
||||||
public KeyserverStatusLiveData(Context context, long masterKeyId) {
|
public KeyserverStatusLiveData(Context context, long masterKeyId) {
|
||||||
super(context, KeyRings.buildGenericKeyRingUri(masterKeyId));
|
super(context, KeyRings.buildGenericKeyRingUri(masterKeyId));
|
||||||
|
|
||||||
this.keyserverStatusDao = KeyserverStatusDao.getInstance(context);
|
this.keyMetadataDao = KeyMetadataDao.create(context);
|
||||||
this.masterKeyId = masterKeyId;
|
this.masterKeyId = masterKeyId;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public KeyserverStatus asyncLoadData() {
|
public KeyMetadata asyncLoadData() {
|
||||||
return keyserverStatusDao.getKeyserverStatus(masterKeyId);
|
return keyMetadataDao.getKeyMetadata(masterKeyId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,11 +25,11 @@ import android.arch.lifecycle.Observer;
|
|||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.support.annotation.Nullable;
|
import android.support.annotation.Nullable;
|
||||||
|
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.KeyserverStatusDao.KeyserverStatus;
|
import org.sufficientlysecure.keychain.model.KeyMetadata;
|
||||||
import org.sufficientlysecure.keychain.ui.keyview.loader.ViewKeyLiveData.KeyserverStatusLiveData;
|
import org.sufficientlysecure.keychain.ui.keyview.loader.ViewKeyLiveData.KeyserverStatusLiveData;
|
||||||
|
|
||||||
|
|
||||||
public class KeyserverStatusPresenter implements Observer<KeyserverStatus> {
|
public class KeyserverStatusPresenter implements Observer<KeyMetadata> {
|
||||||
private final Context context;
|
private final Context context;
|
||||||
private final KeyserverStatusMvpView view;
|
private final KeyserverStatusMvpView view;
|
||||||
|
|
||||||
@@ -46,12 +46,12 @@ public class KeyserverStatusPresenter implements Observer<KeyserverStatus> {
|
|||||||
this.isSecret = isSecret;
|
this.isSecret = isSecret;
|
||||||
}
|
}
|
||||||
|
|
||||||
public LiveData<KeyserverStatus> getLiveDataInstance() {
|
public LiveData<KeyMetadata> getLiveDataInstance() {
|
||||||
return new KeyserverStatusLiveData(context, masterKeyId);
|
return new KeyserverStatusLiveData(context, masterKeyId);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onChanged(@Nullable KeyserverStatus keyserverStatus) {
|
public void onChanged(@Nullable KeyMetadata keyserverStatus) {
|
||||||
if (keyserverStatus == null) {
|
if (keyserverStatus == null) {
|
||||||
view.setDisplayStatusUnknown();
|
view.setDisplayStatusUnknown();
|
||||||
return;
|
return;
|
||||||
@@ -63,7 +63,7 @@ public class KeyserverStatusPresenter implements Observer<KeyserverStatus> {
|
|||||||
} else {
|
} else {
|
||||||
view.setDisplayStatusNotPublished();
|
view.setDisplayStatusNotPublished();
|
||||||
}
|
}
|
||||||
view.setLastUpdated(keyserverStatus.getLastUpdated());
|
view.setLastUpdated(keyserverStatus.last_updated());
|
||||||
} else {
|
} else {
|
||||||
view.setDisplayStatusUnknown();
|
view.setDisplayStatusUnknown();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,28 @@
|
|||||||
|
import java.lang.Boolean;
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS key_metadata (
|
||||||
|
master_key_id INTEGER PRIMARY KEY,
|
||||||
|
last_updated INTEGER AS Date,
|
||||||
|
seen_on_keyservers INTEGER AS Boolean,
|
||||||
|
FOREIGN KEY(master_key_id) REFERENCES keyrings_public(master_key_id) ON DELETE CASCADE
|
||||||
|
);
|
||||||
|
|
||||||
|
selectByMasterKeyId:
|
||||||
|
SELECT *
|
||||||
|
FROM key_metadata
|
||||||
|
WHERE master_key_id = ?;
|
||||||
|
|
||||||
|
deleteAllLastUpdatedTimes:
|
||||||
|
UPDATE key_metadata
|
||||||
|
SET last_updated = null, seen_on_keyservers = null;
|
||||||
|
|
||||||
|
replaceKeyMetadata:
|
||||||
|
REPLACE INTO key_metadata
|
||||||
|
(master_key_id, last_updated, seen_on_keyservers) VALUES (?, ?, ?);
|
||||||
|
|
||||||
|
selectFingerprintsForKeysOlderThan:
|
||||||
|
SELECT fingerprint
|
||||||
|
FROM key_metadata
|
||||||
|
LEFT JOIN keys ON (key_metadata.master_key_id = keys.master_key_id AND keys.rank = 0)
|
||||||
|
WHERE last_updated < ?;
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
CREATE TABLE IF NOT EXISTS keyrings_public (
|
||||||
|
master_key_id INTEGER PRIMARY KEY,
|
||||||
|
key_ring_data BLOB
|
||||||
|
);
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
CREATE TABLE IF NOT EXISTS keys (
|
||||||
|
master_key_id INTEGER,
|
||||||
|
rank INTEGER,
|
||||||
|
key_id INTEGER,
|
||||||
|
key_size INTEGER,
|
||||||
|
key_curve_oid TEXT,
|
||||||
|
algorithm INTEGER,
|
||||||
|
fingerprint BLOB,
|
||||||
|
can_certify INTEGER,
|
||||||
|
can_sign INTEGER,
|
||||||
|
can_encrypt INTEGER,
|
||||||
|
can_authenticate INTEGER,
|
||||||
|
is_revoked INTEGER,
|
||||||
|
has_secret INTEGER,
|
||||||
|
is_secure INTEGER,
|
||||||
|
creation INTEGER,
|
||||||
|
expiry INTEGER,
|
||||||
|
PRIMARY KEY(master_key_id, rank),
|
||||||
|
FOREIGN KEY(master_key_id) REFERENCES
|
||||||
|
keyrings_public(master_key_id) ON DELETE CASCADE
|
||||||
|
);
|
||||||
@@ -16,6 +16,17 @@
|
|||||||
|
|
||||||
package org.sufficientlysecure.keychain.provider;
|
package org.sufficientlysecure.keychain.provider;
|
||||||
|
|
||||||
|
|
||||||
|
import java.io.BufferedInputStream;
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
|
import java.io.Closeable;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.security.Security;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
|
|
||||||
import org.bouncycastle.jce.provider.BouncyCastleProvider;
|
import org.bouncycastle.jce.provider.BouncyCastleProvider;
|
||||||
@@ -47,16 +58,6 @@ import org.sufficientlysecure.keychain.ui.util.KeyFormattingUtils;
|
|||||||
import org.sufficientlysecure.keychain.util.InputData;
|
import org.sufficientlysecure.keychain.util.InputData;
|
||||||
import org.sufficientlysecure.keychain.util.Passphrase;
|
import org.sufficientlysecure.keychain.util.Passphrase;
|
||||||
|
|
||||||
import java.io.BufferedInputStream;
|
|
||||||
import java.io.ByteArrayInputStream;
|
|
||||||
import java.io.ByteArrayOutputStream;
|
|
||||||
import java.io.Closeable;
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileInputStream;
|
|
||||||
import java.net.URL;
|
|
||||||
import java.security.Security;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
|
|
||||||
@RunWith(KeychainTestRunner.class)
|
@RunWith(KeychainTestRunner.class)
|
||||||
public class InteropTest {
|
public class InteropTest {
|
||||||
|
|
||||||
@@ -245,8 +246,7 @@ public class InteropTest {
|
|||||||
KeyWritableRepository helper = new KeyWritableRepository(RuntimeEnvironment.application,
|
KeyWritableRepository helper = new KeyWritableRepository(RuntimeEnvironment.application,
|
||||||
LocalPublicKeyStorage.getInstance(RuntimeEnvironment.application),
|
LocalPublicKeyStorage.getInstance(RuntimeEnvironment.application),
|
||||||
LocalSecretKeyStorage.getInstance(RuntimeEnvironment.application),
|
LocalSecretKeyStorage.getInstance(RuntimeEnvironment.application),
|
||||||
DatabaseNotifyManager.create(RuntimeEnvironment.application),
|
DatabaseNotifyManager.create(RuntimeEnvironment.application)) {
|
||||||
LastUpdateInteractor.create(RuntimeEnvironment.application)) {
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public CachedPublicKeyRing getCachedPublicKeyRing(Uri queryUri) throws PgpKeyNotFoundException {
|
public CachedPublicKeyRing getCachedPublicKeyRing(Uri queryUri) throws PgpKeyNotFoundException {
|
||||||
|
|||||||
Reference in New Issue
Block a user