Replace implementations of Base16, Base32 and Base64 with Guava

I kept the classes in the encoding package and turned them into wrappers for
Guava. I also changed the functions in the Base32 class to take and return
strings insteads if character arrays.
This commit is contained in:
Alexander Bakker 2019-12-26 20:47:28 +01:00
parent 833e75ade1
commit 10ac1af6b0
23 changed files with 88 additions and 254 deletions

View file

@ -21,6 +21,7 @@ android {
targetSdkVersion 29 targetSdkVersion 29
versionCode 28 versionCode 28
versionName "1.1.4" versionName "1.1.4"
multiDexEnabled true
buildConfigField "String", "GIT_HASH", "\"${getGitHash()}\"" buildConfigField "String", "GIT_HASH", "\"${getGitHash()}\""
buildConfigField "String", "GIT_BRANCH", "\"${getGitBranch()}\"" buildConfigField "String", "GIT_BRANCH", "\"${getGitBranch()}\""
} }
@ -72,6 +73,7 @@ dependencies {
implementation 'androidx.preference:preference:1.1.0' implementation 'androidx.preference:preference:1.1.0'
implementation 'com.google.android.material:material:1.0.0' implementation 'com.google.android.material:material:1.0.0'
implementation 'androidx.recyclerview:recyclerview:1.1.0' implementation 'androidx.recyclerview:recyclerview:1.1.0'
implementation 'com.google.guava:guava:28.2-android'
implementation 'com.getbase:floatingactionbutton:1.10.1' implementation 'com.getbase:floatingactionbutton:1.10.1'
implementation 'com.github.apl-devs:appintro:5.1.0' implementation 'com.github.apl-devs:appintro:5.1.0'
implementation 'com.github.avito-tech:krop:0.44' implementation 'com.github.avito-tech:krop:0.44'
@ -91,5 +93,6 @@ dependencies {
annotationProcessor 'androidx.annotation:annotation:1.1.0' annotationProcessor 'androidx.annotation:annotation:1.1.0'
annotationProcessor 'com.github.bumptech.glide:compiler:4.10.0' annotationProcessor 'com.github.bumptech.glide:compiler:4.10.0'
testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.5.2' testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.5.2'
testImplementation 'com.google.guava:guava:28.2-jre"'
testImplementation 'org.junit.jupiter:junit-jupiter-api:5.5.2' testImplementation 'org.junit.jupiter:junit-jupiter-api:5.5.2'
} }

View file

@ -1,7 +1,7 @@
package com.beemdevelopment.aegis.crypto; package com.beemdevelopment.aegis.crypto;
import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.encoding.Hex; import com.beemdevelopment.aegis.encoding.Hex;
import com.beemdevelopment.aegis.encoding.HexException;
import org.json.JSONException; import org.json.JSONException;
import org.json.JSONObject; import org.json.JSONObject;
@ -30,7 +30,7 @@ public class CryptParameters implements Serializable {
return obj; return obj;
} }
public static CryptParameters fromJson(JSONObject obj) throws JSONException, HexException { public static CryptParameters fromJson(JSONObject obj) throws JSONException, EncodingException {
byte[] nonce = Hex.decode(obj.getString("nonce")); byte[] nonce = Hex.decode(obj.getString("nonce"));
byte[] tag = Hex.decode(obj.getString("tag")); byte[] tag = Hex.decode(obj.getString("tag"));
return new CryptParameters(nonce, tag); return new CryptParameters(nonce, tag);

View file

@ -1,155 +1,23 @@
package com.beemdevelopment.aegis.encoding; package com.beemdevelopment.aegis.encoding;
// modified for use in Aegis import com.google.common.io.BaseEncoding;
/* (PD) 2001 The Bitzi Corporation
* Please see http://bitzi.com/publicdomain for more info.
*
* As modified by Patrick Woodworth:
*
* Copyright 2011 Patrick Woodworth
*
* 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.
*/
import java.util.Arrays;
/**
* Base32 - encodes and decodes RFC3548 Base32
* (see http://www.faqs.org/rfcs/rfc3548.html )
*
* @author Robert Kaye
* @author Gordon Mohr
*/
public class Base32 { public class Base32 {
private static final String base32Chars = private static final BaseEncoding _encoding = BaseEncoding.base32().omitPadding();
"ABCDEFGHIJKLMNOPQRSTUVWXYZ234567";
private static final int[] base32Lookup =
{ 0xFF,0xFF,0x1A,0x1B,0x1C,0x1D,0x1E,0x1F, // '0', '1', '2', '3', '4', '5', '6', '7'
0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF, // '8', '9', ':', ';', '<', '=', '>', '?'
0xFF,0x00,0x01,0x02,0x03,0x04,0x05,0x06, // '@', 'A', 'B', 'C', 'D', 'E', 'F', 'G'
0x07,0x08,0x09,0x0A,0x0B,0x0C,0x0D,0x0E, // 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O'
0x0F,0x10,0x11,0x12,0x13,0x14,0x15,0x16, // 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W'
0x17,0x18,0x19,0xFF,0xFF,0xFF,0xFF,0xFF, // 'X', 'Y', 'Z', '[', '\', ']', '^', '_'
0xFF,0x00,0x01,0x02,0x03,0x04,0x05,0x06, // '`', 'a', 'b', 'c', 'd', 'e', 'f', 'g'
0x07,0x08,0x09,0x0A,0x0B,0x0C,0x0D,0x0E, // 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o'
0x0F,0x10,0x11,0x12,0x13,0x14,0x15,0x16, // 'p', 'q', 'r', 's', 't', 'u', 'v', 'w'
0x17,0x18,0x19,0xFF,0xFF,0xFF,0xFF,0xFF // 'x', 'y', 'z', '{', '|', '}', '~', 'DEL'
};
/** private Base32() {
* Encodes byte array to Base32 String.
*
* @param bytes Bytes to encode.
* @return Encoded byte array <code>bytes</code> as a String.
*
*/
public static char[] encode(final byte[] bytes) {
int i = 0, index = 0, digit = 0, j = 0;
int currByte, nextByte;
char[] base32 = new char[(bytes.length + 7) * 8 / 5];
while (i < bytes.length) {
currByte = (bytes[i] >= 0) ? bytes[i] : (bytes[i] + 256); // unsign
/* Is the current digit going to span a byte boundary? */
if (index > 3) {
if ((i + 1) < bytes.length) {
nextByte =
(bytes[i + 1] >= 0) ? bytes[i + 1] : (bytes[i + 1] + 256);
} else {
nextByte = 0;
}
digit = currByte & (0xFF >> index);
index = (index + 5) % 8;
digit <<= index;
digit |= nextByte >> (8 - index);
i++;
} else {
digit = (currByte >> (8 - (index + 5))) & 0x1F;
index = (index + 5) % 8;
if (index == 0)
i++;
}
base32[j++] = base32Chars.charAt(digit);
}
return Arrays.copyOf(base32, j);
} }
/** public static byte[] decode(String s) throws EncodingException {
* Decodes the given Base32 String to a raw byte array. try {
* return _encoding.decode(s.toUpperCase());
* @param base32 } catch (IllegalArgumentException e) {
* @return Decoded <code>base32</code> String as a raw byte array. throw new EncodingException(e);
*/
public static byte[] decode(final char[] base32) throws Base32Exception {
int i, index, lookup, offset, digit;
byte[] bytes = new byte[base32.length * 5 / 8];
for (i = 0, index = 0, offset = 0; i < base32.length; i++) {
// stop decoding when a padding char is encountered
if (base32[i] == '=') {
// make sure the rest is also padding, but don't bother verifying the length
for (int j = i + 1; j < base32.length; j++) {
if (base32[j] != '=') {
throw new Base32Exception("bad padding");
}
}
break;
}
lookup = base32[i] - '0';
digit = decodeDigit(lookup);
if (index <= 3) {
index = (index + 5) % 8;
if (index == 0) {
bytes[offset] |= digit;
offset++;
if (offset >= bytes.length)
break;
} else {
bytes[offset] |= digit << (8 - index);
}
} else {
index = (index + 5) % 8;
bytes[offset] |= (digit >>> index);
offset++;
if (offset >= bytes.length) {
break;
}
bytes[offset] |= digit << (8 - index);
}
} }
return bytes;
} }
private static int decodeDigit(int c) throws Base32Exception { public static String encode(byte[] data) {
/* Skip chars outside the lookup table */ return _encoding.encode(data);
if (c < 0 || c >= base32Lookup.length) {
throw new Base32Exception("char not found in base32 lookup table");
}
int digit = base32Lookup[c];
/* If this digit is not in the table, ignore it */
if (digit == 0xFF) {
throw new Base32Exception("char not found in base32 lookup table");
}
return digit;
} }
} }

View file

@ -1,7 +0,0 @@
package com.beemdevelopment.aegis.encoding;
public class Base32Exception extends Exception {
public Base32Exception(String message) {
super(message);
}
}

View file

@ -1,24 +1,21 @@
package com.beemdevelopment.aegis.encoding; package com.beemdevelopment.aegis.encoding;
import java.nio.charset.StandardCharsets; import com.google.common.io.BaseEncoding;
public class Base64 { public class Base64 {
private static final int _flags = android.util.Base64.NO_WRAP;
private Base64() { private Base64() {
} }
public static byte[] decode(String s) throws Base64Exception { public static byte[] decode(String s) throws EncodingException {
try { try {
return android.util.Base64.decode(s, _flags); return BaseEncoding.base64().decode(s);
} catch (IllegalArgumentException e) { } catch (IllegalArgumentException e) {
throw new Base64Exception(e); throw new EncodingException(e);
} }
} }
public static String encode(byte[] data) { public static String encode(byte[] data) {
byte[] encoded = android.util.Base64.encode(data, _flags); return BaseEncoding.base64().encode(data);
return new String(encoded, StandardCharsets.UTF_8);
} }
} }

View file

@ -1,7 +0,0 @@
package com.beemdevelopment.aegis.encoding;
public class Base64Exception extends Exception {
public Base64Exception(Throwable cause) {
super(cause);
}
}

View file

@ -0,0 +1,9 @@
package com.beemdevelopment.aegis.encoding;
import java.io.IOException;
public class EncodingException extends IOException {
public EncodingException(Throwable cause) {
super(cause);
}
}

View file

@ -1,46 +1,21 @@
package com.beemdevelopment.aegis.encoding; package com.beemdevelopment.aegis.encoding;
// The hexadecimal utility functions in this file were taken and modified from: http://www.docjar.com/html/api/com/sun/xml/internal/bind/DatatypeConverterImpl.java.html import com.google.common.io.BaseEncoding;
// It is licensed under GPLv2 with a classpath exception.
public class Hex { public class Hex {
private Hex() { private Hex() {
} }
private static int hexToBin(char ch) { public static byte[] decode(String s) throws EncodingException {
if ('0' <= ch && ch <= '9') return ch - '0'; try {
if ('A' <= ch && ch <= 'F') return ch - 'A' + 10; return BaseEncoding.base16().decode(s.toUpperCase());
if ('a' <= ch && ch <= 'f') return ch - 'a' + 10; } catch (IllegalArgumentException e) {
return -1; throw new EncodingException(e);
}
private static final char[] hexCode = "0123456789abcdef".toCharArray();
public static byte[] decode(String s) throws HexException {
final int len = s.length();
if (len % 2 != 0)
throw new HexException("hexBinary needs to be even-length: " + s);
byte[] out = new byte[len / 2];
for (int i = 0; i < len; i += 2) {
int h = hexToBin(s.charAt(i));
int l = hexToBin(s.charAt(i + 1));
if (h == -1 || l == -1)
throw new HexException("contains illegal character for hexBinary: " + s);
out[i / 2] = (byte) (h * 16 + l);
} }
return out;
} }
public static String encode(byte[] data) { public static String encode(byte[] data) {
StringBuilder r = new StringBuilder(data.length * 2); return BaseEncoding.base16().lowerCase().encode(data);
for (byte b : data) {
r.append(hexCode[(b >> 4) & 0xF]);
r.append(hexCode[(b & 0xF)]);
}
return r.toString();
} }
} }

View file

@ -1,7 +0,0 @@
package com.beemdevelopment.aegis.encoding;
public class HexException extends Exception {
public HexException(String message) {
super(message);
}
}

View file

@ -2,13 +2,13 @@ package com.beemdevelopment.aegis.importers;
import android.content.Context; import android.content.Context;
import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.otp.OtpInfoException;
import com.beemdevelopment.aegis.vault.VaultEntry; import com.beemdevelopment.aegis.vault.VaultEntry;
import com.beemdevelopment.aegis.vault.VaultFile; import com.beemdevelopment.aegis.vault.VaultFile;
import com.beemdevelopment.aegis.vault.VaultFileCredentials; import com.beemdevelopment.aegis.vault.VaultFileCredentials;
import com.beemdevelopment.aegis.vault.VaultFileException; import com.beemdevelopment.aegis.vault.VaultFileException;
import com.beemdevelopment.aegis.vault.slots.SlotList; import com.beemdevelopment.aegis.vault.slots.SlotList;
import com.beemdevelopment.aegis.encoding.Base64Exception;
import com.beemdevelopment.aegis.otp.OtpInfoException;
import org.json.JSONArray; import org.json.JSONArray;
import org.json.JSONException; import org.json.JSONException;
@ -102,7 +102,7 @@ public class AegisImporter extends DatabaseImporter {
private static VaultEntry convertEntry(JSONObject obj) throws DatabaseImporterEntryException { private static VaultEntry convertEntry(JSONObject obj) throws DatabaseImporterEntryException {
try { try {
return VaultEntry.fromJson(obj); return VaultEntry.fromJson(obj);
} catch (JSONException | OtpInfoException | Base64Exception e) { } catch (JSONException | OtpInfoException | EncodingException e) {
throw new DatabaseImporterEntryException(e, obj.toString()); throw new DatabaseImporterEntryException(e, obj.toString());
} }
} }

View file

@ -10,7 +10,7 @@ import com.beemdevelopment.aegis.crypto.CryptResult;
import com.beemdevelopment.aegis.crypto.CryptoUtils; import com.beemdevelopment.aegis.crypto.CryptoUtils;
import com.beemdevelopment.aegis.vault.VaultEntry; import com.beemdevelopment.aegis.vault.VaultEntry;
import com.beemdevelopment.aegis.encoding.Base32; import com.beemdevelopment.aegis.encoding.Base32;
import com.beemdevelopment.aegis.encoding.Base32Exception; import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.otp.HotpInfo; import com.beemdevelopment.aegis.otp.HotpInfo;
import com.beemdevelopment.aegis.otp.OtpInfo; import com.beemdevelopment.aegis.otp.OtpInfo;
import com.beemdevelopment.aegis.otp.OtpInfoException; import com.beemdevelopment.aegis.otp.OtpInfoException;
@ -201,7 +201,7 @@ public class AndOtpImporter extends DatabaseImporter {
String type = obj.getString("type").toLowerCase(); String type = obj.getString("type").toLowerCase();
String algo = obj.getString("algorithm"); String algo = obj.getString("algorithm");
int digits = obj.getInt("digits"); int digits = obj.getInt("digits");
byte[] secret = Base32.decode(obj.getString("secret").toCharArray()); byte[] secret = Base32.decode(obj.getString("secret"));
OtpInfo info; OtpInfo info;
switch (type) { switch (type) {
@ -230,7 +230,7 @@ public class AndOtpImporter extends DatabaseImporter {
} }
return new VaultEntry(info, name, issuer); return new VaultEntry(info, name, issuer);
} catch (DatabaseImporterException | Base32Exception | OtpInfoException | JSONException e) { } catch (DatabaseImporterException | EncodingException | OtpInfoException | JSONException e) {
throw new DatabaseImporterEntryException(e, obj.toString()); throw new DatabaseImporterEntryException(e, obj.toString());
} }
} }

View file

@ -5,7 +5,7 @@ import android.util.Xml;
import com.beemdevelopment.aegis.vault.VaultEntry; import com.beemdevelopment.aegis.vault.VaultEntry;
import com.beemdevelopment.aegis.encoding.Base32; import com.beemdevelopment.aegis.encoding.Base32;
import com.beemdevelopment.aegis.encoding.Base32Exception; import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.otp.OtpInfo; import com.beemdevelopment.aegis.otp.OtpInfo;
import com.beemdevelopment.aegis.otp.OtpInfoException; import com.beemdevelopment.aegis.otp.OtpInfoException;
import com.beemdevelopment.aegis.otp.TotpInfo; import com.beemdevelopment.aegis.otp.TotpInfo;
@ -97,12 +97,12 @@ public class AuthyImporter extends DatabaseImporter {
sanitizeEntryInfo(authyEntryInfo); sanitizeEntryInfo(authyEntryInfo);
int digits = entry.getInt("digits"); int digits = entry.getInt("digits");
byte[] secret = Base32.decode(entry.getString("decryptedSecret").toCharArray()); byte[] secret = Base32.decode(entry.getString("decryptedSecret"));
OtpInfo info = new TotpInfo(secret, "SHA1", digits, 30); OtpInfo info = new TotpInfo(secret, "SHA1", digits, 30);
return new VaultEntry(info, authyEntryInfo.Name, authyEntryInfo.Issuer); return new VaultEntry(info, authyEntryInfo.Name, authyEntryInfo.Issuer);
} catch (OtpInfoException | JSONException | Base32Exception e) { } catch (OtpInfoException | JSONException | EncodingException e) {
throw new DatabaseImporterEntryException(e, entry.toString()); throw new DatabaseImporterEntryException(e, entry.toString());
} }
} }

View file

@ -7,7 +7,7 @@ import android.database.sqlite.SQLiteException;
import com.beemdevelopment.aegis.vault.VaultEntry; import com.beemdevelopment.aegis.vault.VaultEntry;
import com.beemdevelopment.aegis.encoding.Base32; import com.beemdevelopment.aegis.encoding.Base32;
import com.beemdevelopment.aegis.encoding.Base32Exception; import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.otp.HotpInfo; import com.beemdevelopment.aegis.otp.HotpInfo;
import com.beemdevelopment.aegis.otp.OtpInfo; import com.beemdevelopment.aegis.otp.OtpInfo;
import com.beemdevelopment.aegis.otp.OtpInfoException; import com.beemdevelopment.aegis.otp.OtpInfoException;
@ -104,7 +104,7 @@ public class GoogleAuthImporter extends DatabaseImporter {
private static VaultEntry convertEntry(Entry entry) throws DatabaseImporterEntryException { private static VaultEntry convertEntry(Entry entry) throws DatabaseImporterEntryException {
try { try {
byte[] secret = Base32.decode(entry.getSecret().toCharArray()); byte[] secret = Base32.decode(entry.getSecret());
OtpInfo info; OtpInfo info;
switch (entry.getType()) { switch (entry.getType()) {
@ -125,7 +125,7 @@ public class GoogleAuthImporter extends DatabaseImporter {
} }
return new VaultEntry(info, name, entry.getIssuer()); return new VaultEntry(info, name, entry.getIssuer());
} catch (Base32Exception | OtpInfoException | DatabaseImporterException e) { } catch (EncodingException | OtpInfoException | DatabaseImporterException e) {
throw new DatabaseImporterEntryException(e, entry.toString()); throw new DatabaseImporterEntryException(e, entry.toString());
} }
} }

View file

@ -3,9 +3,9 @@ package com.beemdevelopment.aegis.importers;
import android.content.Context; import android.content.Context;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.vault.VaultEntry; import com.beemdevelopment.aegis.vault.VaultEntry;
import com.beemdevelopment.aegis.encoding.Base64; import com.beemdevelopment.aegis.encoding.Base64;
import com.beemdevelopment.aegis.encoding.Base64Exception;
import com.beemdevelopment.aegis.otp.OtpInfoException; import com.beemdevelopment.aegis.otp.OtpInfoException;
import com.beemdevelopment.aegis.otp.SteamInfo; import com.beemdevelopment.aegis.otp.SteamInfo;
import com.beemdevelopment.aegis.util.ByteInputStream; import com.beemdevelopment.aegis.util.ByteInputStream;
@ -82,7 +82,7 @@ public class SteamImporter extends DatabaseImporter {
String account = obj.getString("account_name"); String account = obj.getString("account_name");
return new VaultEntry(info, account, "Steam"); return new VaultEntry(info, account, "Steam");
} catch (JSONException | Base64Exception | OtpInfoException e) { } catch (JSONException | EncodingException | OtpInfoException e) {
throw new DatabaseImporterEntryException(e, obj.toString()); throw new DatabaseImporterEntryException(e, obj.toString());
} }
} }

View file

@ -3,7 +3,7 @@ package com.beemdevelopment.aegis.otp;
import android.net.Uri; import android.net.Uri;
import com.beemdevelopment.aegis.encoding.Base32; import com.beemdevelopment.aegis.encoding.Base32;
import com.beemdevelopment.aegis.encoding.Base32Exception; import com.beemdevelopment.aegis.encoding.EncodingException;
public class GoogleAuthInfo { public class GoogleAuthInfo {
private OtpInfo _info; private OtpInfo _info;
@ -71,8 +71,8 @@ public class GoogleAuthInfo {
// decode secret // decode secret
byte[] secret; byte[] secret;
try { try {
secret = Base32.decode(encodedSecret.toCharArray()); secret = Base32.decode(encodedSecret);
} catch (Base32Exception e) { } catch (EncodingException e) {
throw new GoogleAuthInfoException("bad secret", e); throw new GoogleAuthInfoException("bad secret", e);
} }

View file

@ -1,7 +1,7 @@
package com.beemdevelopment.aegis.otp; package com.beemdevelopment.aegis.otp;
import com.beemdevelopment.aegis.encoding.Base32; import com.beemdevelopment.aegis.encoding.Base32;
import com.beemdevelopment.aegis.encoding.Base32Exception; import com.beemdevelopment.aegis.encoding.EncodingException;
import org.json.JSONException; import org.json.JSONException;
import org.json.JSONObject; import org.json.JSONObject;
@ -93,7 +93,7 @@ public abstract class OtpInfo implements Serializable {
OtpInfo info; OtpInfo info;
try { try {
byte[] secret = Base32.decode(obj.getString("secret").toCharArray()); byte[] secret = Base32.decode(obj.getString("secret"));
String algo = obj.getString("algo"); String algo = obj.getString("algo");
int digits = obj.getInt("digits"); int digits = obj.getInt("digits");
@ -110,7 +110,7 @@ public abstract class OtpInfo implements Serializable {
default: default:
throw new OtpInfoException("unsupported otp type: " + type); throw new OtpInfoException("unsupported otp type: " + type);
} }
} catch (Base32Exception | JSONException e) { } catch (EncodingException | JSONException e) {
throw new OtpInfoException(e); throw new OtpInfoException(e);
} }

View file

@ -22,12 +22,18 @@ import android.widget.RelativeLayout;
import android.widget.Spinner; import android.widget.Spinner;
import android.widget.TableRow; import android.widget.TableRow;
import androidx.annotation.ArrayRes;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.app.ActionBar;
import androidx.appcompat.app.AlertDialog;
import com.amulyakhare.textdrawable.TextDrawable; import com.amulyakhare.textdrawable.TextDrawable;
import com.avito.android.krop.KropView; import com.avito.android.krop.KropView;
import com.beemdevelopment.aegis.R; import com.beemdevelopment.aegis.R;
import com.beemdevelopment.aegis.vault.VaultEntry; import com.beemdevelopment.aegis.vault.VaultEntry;
import com.beemdevelopment.aegis.encoding.Base32; import com.beemdevelopment.aegis.encoding.Base32;
import com.beemdevelopment.aegis.encoding.Base32Exception; import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.helpers.EditTextHelper; import com.beemdevelopment.aegis.helpers.EditTextHelper;
import com.beemdevelopment.aegis.helpers.SpinnerHelper; import com.beemdevelopment.aegis.helpers.SpinnerHelper;
import com.beemdevelopment.aegis.helpers.TextDrawableHelper; import com.beemdevelopment.aegis.helpers.TextDrawableHelper;
@ -36,11 +42,11 @@ import com.beemdevelopment.aegis.otp.OtpInfo;
import com.beemdevelopment.aegis.otp.OtpInfoException; import com.beemdevelopment.aegis.otp.OtpInfoException;
import com.beemdevelopment.aegis.otp.SteamInfo; import com.beemdevelopment.aegis.otp.SteamInfo;
import com.beemdevelopment.aegis.otp.TotpInfo; import com.beemdevelopment.aegis.otp.TotpInfo;
import com.beemdevelopment.aegis.util.Cloner;
import com.bumptech.glide.Glide; import com.bumptech.glide.Glide;
import com.bumptech.glide.load.engine.DiskCacheStrategy; import com.bumptech.glide.load.engine.DiskCacheStrategy;
import com.bumptech.glide.request.target.CustomTarget; import com.bumptech.glide.request.target.CustomTarget;
import com.bumptech.glide.request.transition.Transition; import com.bumptech.glide.request.transition.Transition;
import com.beemdevelopment.aegis.util.Cloner;
import java.io.ByteArrayOutputStream; import java.io.ByteArrayOutputStream;
import java.text.Collator; import java.text.Collator;
@ -49,11 +55,6 @@ import java.util.List;
import java.util.TreeSet; import java.util.TreeSet;
import java.util.concurrent.atomic.AtomicReference; import java.util.concurrent.atomic.AtomicReference;
import androidx.annotation.ArrayRes;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.app.ActionBar;
import androidx.appcompat.app.AlertDialog;
import de.hdodenhof.circleimageview.CircleImageView; import de.hdodenhof.circleimageview.CircleImageView;
public class EditEntryActivity extends AegisActivity { public class EditEntryActivity extends AegisActivity {
@ -163,8 +164,8 @@ public class EditEntryActivity extends AegisActivity {
byte[] secretBytes = _origEntry.getInfo().getSecret(); byte[] secretBytes = _origEntry.getInfo().getSecret();
if (secretBytes != null) { if (secretBytes != null) {
char[] secretChars = Base32.encode(secretBytes); String secretString = Base32.encode(secretBytes);
_textSecret.setText(secretChars, 0, secretChars.length); _textSecret.setText(secretString);
} }
String type = _origEntry.getInfo().getType(); String type = _origEntry.getInfo().getType();
@ -458,8 +459,11 @@ public class EditEntryActivity extends AegisActivity {
byte[] secret; byte[] secret;
try { try {
secret = Base32.decode(EditTextHelper.getEditTextChars(_textSecret, true)); secret = Base32.decode(new String(EditTextHelper.getEditTextChars(_textSecret, true)));
} catch (Base32Exception e) { if (secret.length == 0) {
throw new ParseException("Secret cannot be empty");
}
} catch (EncodingException e) {
throw new ParseException("Secret is not valid base32."); throw new ParseException("Secret is not valid base32.");
} }

View file

@ -1,6 +1,6 @@
package com.beemdevelopment.aegis.vault; package com.beemdevelopment.aegis.vault;
import com.beemdevelopment.aegis.encoding.Base64Exception; import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.otp.OtpInfoException; import com.beemdevelopment.aegis.otp.OtpInfoException;
import com.beemdevelopment.aegis.util.UUIDMap; import com.beemdevelopment.aegis.util.UUIDMap;
@ -43,7 +43,7 @@ public class Vault {
VaultEntry entry = VaultEntry.fromJson(array.getJSONObject(i)); VaultEntry entry = VaultEntry.fromJson(array.getJSONObject(i));
entries.add(entry); entries.add(entry);
} }
} catch (Base64Exception | OtpInfoException | JSONException e) { } catch (EncodingException | OtpInfoException | JSONException e) {
throw new VaultException(e); throw new VaultException(e);
} }

View file

@ -1,7 +1,7 @@
package com.beemdevelopment.aegis.vault; package com.beemdevelopment.aegis.vault;
import com.beemdevelopment.aegis.encoding.Base64; import com.beemdevelopment.aegis.encoding.Base64;
import com.beemdevelopment.aegis.encoding.Base64Exception; import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.otp.GoogleAuthInfo; import com.beemdevelopment.aegis.otp.GoogleAuthInfo;
import com.beemdevelopment.aegis.otp.OtpInfo; import com.beemdevelopment.aegis.otp.OtpInfo;
import com.beemdevelopment.aegis.otp.OtpInfoException; import com.beemdevelopment.aegis.otp.OtpInfoException;
@ -60,7 +60,7 @@ public class VaultEntry extends UUIDMap.Value {
return obj; return obj;
} }
public static VaultEntry fromJson(JSONObject obj) throws JSONException, OtpInfoException, Base64Exception { public static VaultEntry fromJson(JSONObject obj) throws JSONException, OtpInfoException, EncodingException {
// if there is no uuid, generate a new one // if there is no uuid, generate a new one
UUID uuid; UUID uuid;
if (!obj.has("uuid")) { if (!obj.has("uuid")) {

View file

@ -3,11 +3,10 @@ package com.beemdevelopment.aegis.vault;
import com.beemdevelopment.aegis.crypto.CryptParameters; import com.beemdevelopment.aegis.crypto.CryptParameters;
import com.beemdevelopment.aegis.crypto.CryptResult; import com.beemdevelopment.aegis.crypto.CryptResult;
import com.beemdevelopment.aegis.crypto.MasterKeyException; import com.beemdevelopment.aegis.crypto.MasterKeyException;
import com.beemdevelopment.aegis.encoding.Base64;
import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.vault.slots.SlotList; import com.beemdevelopment.aegis.vault.slots.SlotList;
import com.beemdevelopment.aegis.vault.slots.SlotListException; import com.beemdevelopment.aegis.vault.slots.SlotListException;
import com.beemdevelopment.aegis.encoding.Base64;
import com.beemdevelopment.aegis.encoding.Base64Exception;
import com.beemdevelopment.aegis.encoding.HexException;
import org.json.JSONException; import org.json.JSONException;
import org.json.JSONObject; import org.json.JSONObject;
@ -95,7 +94,7 @@ public class VaultFile {
byte[] bytes = Base64.decode((String) _content); byte[] bytes = Base64.decode((String) _content);
CryptResult result = creds.decrypt(bytes, _header.getParams()); CryptResult result = creds.decrypt(bytes, _header.getParams());
return new JSONObject(new String(result.getData(), StandardCharsets.UTF_8)); return new JSONObject(new String(result.getData(), StandardCharsets.UTF_8));
} catch (MasterKeyException | JSONException | Base64Exception e) { } catch (MasterKeyException | JSONException | EncodingException e) {
throw new VaultFileException(e); throw new VaultFileException(e);
} }
} }
@ -136,7 +135,7 @@ public class VaultFile {
SlotList slots = SlotList.fromJson(obj.getJSONArray("slots")); SlotList slots = SlotList.fromJson(obj.getJSONArray("slots"));
CryptParameters params = CryptParameters.fromJson(obj.getJSONObject("params")); CryptParameters params = CryptParameters.fromJson(obj.getJSONObject("params"));
return new Header(slots, params); return new Header(slots, params);
} catch (SlotListException | JSONException | HexException e) { } catch (SlotListException | JSONException | EncodingException e) {
throw new VaultFileException(e); throw new VaultFileException(e);
} }
} }

View file

@ -5,8 +5,8 @@ import com.beemdevelopment.aegis.crypto.CryptResult;
import com.beemdevelopment.aegis.crypto.CryptoUtils; import com.beemdevelopment.aegis.crypto.CryptoUtils;
import com.beemdevelopment.aegis.crypto.MasterKey; import com.beemdevelopment.aegis.crypto.MasterKey;
import com.beemdevelopment.aegis.crypto.SCryptParameters; import com.beemdevelopment.aegis.crypto.SCryptParameters;
import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.encoding.Hex; import com.beemdevelopment.aegis.encoding.Hex;
import com.beemdevelopment.aegis.encoding.HexException;
import com.beemdevelopment.aegis.util.UUIDMap; import com.beemdevelopment.aegis.util.UUIDMap;
import org.json.JSONException; import org.json.JSONException;
@ -144,7 +144,7 @@ public abstract class Slot extends UUIDMap.Value {
default: default:
throw new SlotException("unrecognized slot type"); throw new SlotException("unrecognized slot type");
} }
} catch (JSONException | HexException e) { } catch (JSONException | EncodingException e) {
throw new SlotException(e); throw new SlotException(e);
} }

View file

@ -2,8 +2,8 @@ package com.beemdevelopment.aegis;
import com.beemdevelopment.aegis.crypto.CryptoUtils; import com.beemdevelopment.aegis.crypto.CryptoUtils;
import com.beemdevelopment.aegis.crypto.SCryptParameters; import com.beemdevelopment.aegis.crypto.SCryptParameters;
import com.beemdevelopment.aegis.encoding.EncodingException;
import com.beemdevelopment.aegis.encoding.Hex; import com.beemdevelopment.aegis.encoding.Hex;
import com.beemdevelopment.aegis.encoding.HexException;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
@ -15,7 +15,7 @@ import static org.junit.jupiter.api.Assertions.*;
public class SCryptTest { public class SCryptTest {
@Test @Test
public void testTrailingNullCollision() throws HexException { public void testTrailingNullCollision() throws EncodingException {
byte[] salt = new byte[0]; byte[] salt = new byte[0];
SCryptParameters params = new SCryptParameters( SCryptParameters params = new SCryptParameters(
CryptoUtils.CRYPTO_SCRYPT_N, CryptoUtils.CRYPTO_SCRYPT_N,

View file

@ -2,14 +2,14 @@ package com.beemdevelopment.aegis;
import com.beemdevelopment.aegis.crypto.otp.OTP; import com.beemdevelopment.aegis.crypto.otp.OTP;
import com.beemdevelopment.aegis.crypto.otp.TOTP; import com.beemdevelopment.aegis.crypto.otp.TOTP;
import com.beemdevelopment.aegis.encoding.HexException;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import java.security.InvalidKeyException; import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException; import java.security.NoSuchAlgorithmException;
import static org.junit.jupiter.api.Assertions.*; import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.fail;
public class TOTPTest { public class TOTPTest {
private static class Vector { private static class Vector {
@ -66,7 +66,7 @@ public class TOTPTest {
}; };
@Test @Test
public void vectorsMatch() throws NoSuchAlgorithmException, InvalidKeyException, HexException { public void vectorsMatch() throws NoSuchAlgorithmException, InvalidKeyException {
for (Vector vector : _vectors) { for (Vector vector : _vectors) {
byte[] seed; byte[] seed;