package org.sufficientlysecure.keychain.provider; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import android.content.Context; import okhttp3.internal.Util; class LocalPublicKeyStorage { private static final String FORMAT_STR_PUBLIC_KEY = "0x%016x.pub"; private static final String PUBLIC_KEYS_DIR_NAME = "public_keys"; private final File localPublicKeysDir; public static LocalPublicKeyStorage getInstance(Context context) { File localPublicKeysDir = new File(context.getFilesDir(), PUBLIC_KEYS_DIR_NAME); return new LocalPublicKeyStorage(localPublicKeysDir); } private LocalPublicKeyStorage(File localPublicKeysDir) { this.localPublicKeysDir = localPublicKeysDir; } private File getPublicKeyFile(long masterKeyId) throws IOException { if (!localPublicKeysDir.exists()) { localPublicKeysDir.mkdir(); } if (!localPublicKeysDir.isDirectory()) { throw new IOException("Failed creating public key directory!"); } String keyFilename = String.format(FORMAT_STR_PUBLIC_KEY, masterKeyId); return new File(localPublicKeysDir, keyFilename); } void writePublicKey(long masterKeyId, byte[] encoded) throws IOException { File publicKeyFile = getPublicKeyFile(masterKeyId); FileOutputStream fileOutputStream = new FileOutputStream(publicKeyFile); try { fileOutputStream.write(encoded); } finally { Util.closeQuietly(fileOutputStream); } } byte[] readPublicKey(long masterKeyId) throws IOException { File publicKeyFile = getPublicKeyFile(masterKeyId); try { FileInputStream fileInputStream = new FileInputStream(publicKeyFile); return readIntoByteArray(fileInputStream); } catch (FileNotFoundException e) { return null; } } private static byte[] readIntoByteArray(FileInputStream fileInputStream) throws IOException { ByteArrayOutputStream baos = new ByteArrayOutputStream(); byte[] buf = new byte[128]; int bytesRead; while ((bytesRead = fileInputStream.read(buf)) != -1) { baos.write(buf, 0, bytesRead); } return baos.toByteArray(); } void deletePublicKey(long masterKeyId) throws IOException { File publicKeyFile = getPublicKeyFile(masterKeyId); if (publicKeyFile.exists()) { boolean deleteSuccess = publicKeyFile.delete(); if (!deleteSuccess) { throw new IOException("File exists, but could not be deleted!"); } } } }