Я хочу реализовать шифрование RSA с открытым и закрытым ключом в Android. Реализация, которую я сделал, как показано ниже.
public static String enccriptData(String dataToEncrypt)
{
try {
String publicExponentString = "AQAB";
String PUBLIC_KEY = "BgIAAACkAABSU0ExAAQAAAEAAQBJGj09Gbyl7BS/8MytvjBUUfaktW984VHHW4lSI9y2OwaeOq4qqSD6IOHU9HL/QtwZ+wELq28eAOQSnr11hifMf6zWjIsCBHOEpLNJjL3wxjl7dUBEGMJOeZj2rmcf8v7lP/rpAtO/G8wKXhAIKLIFxcChkXwQKbQbHQ/FtX2bwg=="
byte[] modulusBytes = Base64.decodeBase64(PUBLIC_KEY);
byte[] exponentBytes = Base64.decodeBase64(publicExponentString);
BigInteger modulus = new BigInteger(1, modulusBytes);
BigInteger publicExponent = new BigInteger(1, exponentBytes);
RSAPublicKeySpec rsaPubKey = new RSAPublicKeySpec(modulus, publicExponent);
KeyFactory fact = null;
fact = KeyFactory.getInstance("RSA");
PublicKey pubKey = fact.generatePublic(rsaPubKey);
Cipher cipher = Cipher.getInstance("RSA/ECB/PKCS1PADDING");
cipher.init(Cipher.ENCRYPT_MODE, pubKey);
String token = dataToEncrypt + ":" + String.valueOf(getCurrentDateTimeTicks());
byte[] plainBytes = token.getBytes("UTF-16LE");
byte[] cipherData = cipher.doFinal(plainBytes);
String encryptedStringBase64 = Base64.encodeBase64String(cipherData);
return encryptedStringBase64;
}catch (Exception e){
e.printStackTrace();
}
return null;
}
private static long getCurrentDateTimeTicks(){
long TICKS_AT_EPOCH = 621355968000000000L;
long tick = System.currentTimeMillis()*10000 + TICKS_AT_EPOCH;
return tick;
}
Это прекрасно работает, когда я запускаю этот код в виде файла Java. Он шифрует данные, как и ожидалось. Но, когда я использую этот код в приложении для Android, он выдает ниже ошибку.
java.lang.RuntimeException: error:03000068:bignum routines:OPENSSL_internal:CALLED_WITH_EVEN_MODULUS
at com.android.org.conscrypt.NativeCrypto.RSA_public_encrypt(Native Method)
at com.android.org.conscrypt.OpenSSLCipherRSA$DirectRSA.doCryptoOperation(OpenSSLCipherRSA.java:371)
at com.android.org.conscrypt.OpenSSLCipherRSA.engineDoFinal(OpenSSLCipherRSA.java:292)
at javax.crypto.Cipher.doFinal(Cipher.java:1741)
Эквивалентный код C # приведен ниже
public string GenerateToken()
{
var completeToken = string.Format("{0}:{1}", _dataToEncrypt, DateTime.UtcNow.Ticks);
var encryptedToken = EncryptionHelper.Encrypt(_publicKey, completeToken);
var token = Convert.ToBase64String(encryptedToken);
return token;
}
public static byte[] Encrypt(string publicKey, string data)
{
var cspParams = new CspParameters { ProviderType = 1 };
var rsaProvider = new RSACryptoServiceProvider(cspParams);
rsaProvider.ImportCspBlob(Convert.FromBase64String(publicKey));
var plainBytes = Encoding.UTF8.GetBytes(data);
var encryptedBytes = rsaProvider.Encrypt(plainBytes, false);
return encryptedBytes;
}
Ссылка
Перевод C # RSACryptoServiceProvider в код JAVA