/* * Copyright (C) 2012-2014 Dominik Schürmann * Copyright (C) 2010-2014 Thialfihar * * 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 . */ package org.sufficientlysecure.keychain.pgp; import android.content.Context; import android.content.pm.PackageInfo; import android.content.pm.PackageManager.NameNotFoundException; import org.sufficientlysecure.keychain.Constants; import org.sufficientlysecure.keychain.R; import org.sufficientlysecure.keychain.util.Log; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.RandomAccessFile; import java.security.SecureRandom; import java.util.regex.Pattern; public class PgpHelper { public static final Pattern PGP_MESSAGE = Pattern.compile( ".*?(-----BEGIN PGP MESSAGE-----.*?-----END PGP MESSAGE-----).*", Pattern.DOTALL); public static final Pattern PGP_CLEARTEXT_SIGNATURE = Pattern .compile(".*?(-----BEGIN PGP SIGNED MESSAGE-----.*?-----" + "BEGIN PGP SIGNATURE-----.*?-----END PGP SIGNATURE-----).*", Pattern.DOTALL); public static final Pattern PGP_PUBLIC_KEY = Pattern.compile( ".*?(-----BEGIN PGP PUBLIC KEY BLOCK-----.*?-----END PGP PUBLIC KEY BLOCK-----).*", Pattern.DOTALL); public static String getVersion(Context context) { String version; try { PackageInfo pi = context.getPackageManager().getPackageInfo(Constants.PACKAGE_NAME, 0); version = pi.versionName; return version; } catch (NameNotFoundException e) { Log.e(Constants.TAG, "Version could not be retrieved!", e); return "0.0"; } } public static String getFullVersion(Context context) { return "OpenKeychain v" + getVersion(context); } // public static final class content { // public static final int unknown = 0; // public static final int encrypted_data = 1; // public static final int keys = 2; // } // // public static int getStreamContent(Context context, InputStream inStream) throws IOException { // // InputStream in = PGPUtil.getDecoderStream(inStream); // PGPObjectFactory pgpF = new PGPObjectFactory(in); // Object object = pgpF.nextObject(); // while (object != null) { // if (object instanceof PGPPublicKeyRing || object instanceof PGPSecretKeyRing) { // return Id.content.keys; // } else if (object instanceof PGPEncryptedDataList) { // return Id.content.encrypted_data; // } // object = pgpF.nextObject(); // } // // return Id.content.unknown; // } /** * Generate a random filename * * @param length * @return */ public static String generateRandomFilename(int length) { SecureRandom random = new SecureRandom(); byte bytes[] = new byte[length]; random.nextBytes(bytes); String result = ""; for (int i = 0; i < length; ++i) { int v = (bytes[i] + 256) % 64; if (v < 10) { result += (char) ('0' + v); } else if (v < 36) { result += (char) ('A' + v - 10); } else if (v < 62) { result += (char) ('a' + v - 36); } else if (v == 62) { result += '_'; } else if (v == 63) { result += '.'; } } return result; } /** * Go once through stream to get length of stream. The length is later used to display progress * when encrypting/decrypting * * @param in * @return * @throws IOException */ public static long getLengthOfStream(InputStream in) throws IOException { long size = 0; long n = 0; byte dummy[] = new byte[0x10000]; while ((n = in.read(dummy)) > 0) { size += n; } return size; } /** * Deletes file securely by overwriting it with random data before deleting it. *

* TODO: Does this really help on flash storage? * * @param context * @param progressable * @param file * @throws IOException */ public static void deleteFileSecurely(Context context, Progressable progressable, File file) throws IOException { long length = file.length(); SecureRandom random = new SecureRandom(); RandomAccessFile raf = new RandomAccessFile(file, "rws"); raf.seek(0); raf.getFilePointer(); byte[] data = new byte[1 << 16]; int pos = 0; String msg = context.getString(R.string.progress_deleting_securely, file.getName()); while (pos < length) { if (progressable != null) { progressable.setProgress(msg, (int) (100 * pos / length), 100); } random.nextBytes(data); raf.write(data); pos += data.length; } raf.close(); file.delete(); } }