com.github.DNAProject.sdk.manager.ECIES Maven / Gradle / Ivy
/*
* Copyright (C) 2018 The DNA Authors
* This file is part of The DNA library.
*
* The DNA is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* The DNA 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 Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with The DNA. If not, see .
*
*/
package com.github.DNAProject.sdk.manager;
import com.github.DNAProject.account.Account;
import com.github.DNAProject.common.ErrorCode;
import com.github.DNAProject.common.Helper;
import com.github.DNAProject.crypto.KeyType;
import com.github.DNAProject.crypto.SignatureScheme;
import org.bouncycastle.crypto.AsymmetricCipherKeyPair;
import org.bouncycastle.crypto.Digest;
import org.bouncycastle.crypto.digests.SHA1Digest;
import org.bouncycastle.crypto.digests.SHA256Digest;
import org.bouncycastle.crypto.generators.KDF2BytesGenerator;
import org.bouncycastle.crypto.kems.ECIESKeyEncapsulation;
import org.bouncycastle.crypto.params.ECDomainParameters;
import org.bouncycastle.crypto.params.ECPrivateKeyParameters;
import org.bouncycastle.crypto.params.ECPublicKeyParameters;
import org.bouncycastle.crypto.params.KeyParameter;
import org.bouncycastle.jcajce.provider.asymmetric.ec.BCECPrivateKey;
import org.bouncycastle.jcajce.provider.asymmetric.ec.BCECPublicKey;
import org.bouncycastle.jce.ECNamedCurveTable;
import org.bouncycastle.jce.spec.ECNamedCurveParameterSpec;
import org.bouncycastle.util.encoders.Hex;
import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import java.security.Key;
import java.security.SecureRandom;
public class ECIES {
public static KeyType keyType = KeyType.ECDSA;
public static Object[] curveParaSpec = new Object[]{"P-256"};
public static SignatureScheme signatureScheme = SignatureScheme.SHA256WITHECDSA;
public static Digest defaultDigest = new SHA256Digest();
public ECIES(Digest dig){
defaultDigest = dig;
}
public static void setDigest(Digest dig){
defaultDigest = dig;
}
public static String[] Encrypt(String pubkey, byte[] msg) throws Exception{
return Encrypt(pubkey, msg, 32);
}
//keylen: 16/24/32
public static String[] Encrypt(String pubkey, byte[] msg, int keylen) throws Exception{
try {
com.github.DNAProject.account.Account account = new com.github.DNAProject.account.Account(false, Helper.hexToBytes(pubkey));
ECNamedCurveParameterSpec spec = ECNamedCurveTable.getParameterSpec((String) curveParaSpec[0]);
ECDomainParameters ecDomain = new ECDomainParameters(spec.getCurve(), spec.getG(), spec.getN());
AsymmetricCipherKeyPair keys = new AsymmetricCipherKeyPair(
new ECPublicKeyParameters(((BCECPublicKey) account.getPublicKey()).getQ(), ecDomain), null);
byte[] out = new byte[(ecDomain.getCurve().getFieldSize() / 8) * 2 + 1];
ECIESKeyEncapsulation kem = new ECIESKeyEncapsulation(new KDF2BytesGenerator(defaultDigest), new SecureRandom());
KeyParameter key1;
kem.init(keys.getPublic());
key1 = (KeyParameter) kem.encrypt(out, keylen); //AES key = key1 (is encrypted in out)
byte[] IV = Hex.decode(getRandomString(keylen)); //choose random IV of length = keylen
byte[] ciphertext;
try {
Cipher en = Cipher.getInstance("AES/CBC/PKCS7Padding", "BC");
Key key = new SecretKeySpec(key1.getKey(), "AES");
en.init(Cipher.ENCRYPT_MODE, key, new IvParameterSpec(IV));
ciphertext = en.doFinal(msg);
} catch (Exception e) {
throw new Exception("AES failed initialisation - " + e.toString(), e);
}
//(IV, out, ciphertext)
return new String[]{Helper.toHexString(IV), Helper.toHexString(out), Helper.toHexString(ciphertext)};
} catch (Exception e) {
throw new Exception("Encrypt failed initialisation - " + e.toString(), e);
}
}
public static byte[] Decrypt(Account account, String[] params) throws Exception {
if (params.length != 3) {
throw new Exception(ErrorCode.ParamError);
}
return Decrypt(account.serializePrivateKey(), Helper.hexToBytes(params[0]), Helper.hexToBytes(params[1]), Helper.hexToBytes(params[2]), 32);
}
public static byte[] Decrypt(String prikey, String[] params) throws Exception {
if (params.length != 3) {
throw new Exception(ErrorCode.ParamError);
}
return Decrypt(Helper.hexToBytes(prikey), Helper.hexToBytes(params[0]), Helper.hexToBytes(params[1]), Helper.hexToBytes(params[2]), 32);
}
public static byte[] Decrypt(byte[] prikey, byte[] IV, byte[] key_cxt, byte[] ciphertext) throws Exception {
return Decrypt(prikey, IV, key_cxt, ciphertext, 32);
}
public static byte[] Decrypt(byte[] prikey, byte[] IV, byte[] key_cxt, byte[] ciphertext, int keylen) throws Exception {
try {
com.github.DNAProject.account.Account account = new com.github.DNAProject.account.Account(prikey, signatureScheme);
ECNamedCurveParameterSpec spec = ECNamedCurveTable.getParameterSpec((String) curveParaSpec[0]);
ECDomainParameters ecDomain = new ECDomainParameters(spec.getCurve(), spec.getG(), spec.getN());
AsymmetricCipherKeyPair keys = new AsymmetricCipherKeyPair(
null,
new ECPrivateKeyParameters(((BCECPrivateKey) account.getPrivateKey()).getD(), ecDomain));
byte[] out = key_cxt;
ECIESKeyEncapsulation kem = new ECIESKeyEncapsulation(new KDF2BytesGenerator(defaultDigest), new SecureRandom());
KeyParameter key1;
kem.init(keys.getPrivate());
key1 = (KeyParameter) kem.decrypt(out, keylen);
byte[] plaintext;
try {
Cipher dec = Cipher.getInstance("AES/CBC/PKCS7Padding", "BC");
Key key = new SecretKeySpec(key1.getKey(), "AES");
dec.init(Cipher.DECRYPT_MODE, key, new IvParameterSpec(IV));
plaintext = dec.doFinal(ciphertext);
} catch (Exception e) {
throw new Exception("AES failed initialisation - " + e.toString(), e);
}
return plaintext;
} catch (Exception e) {
throw new Exception("AES failed initialisation - " + e.toString(), e);
}
}
public static String getRandomString(int length) {
String KeyStr = "ABCDEF0123456789";
StringBuffer sb = new StringBuffer();
int len = KeyStr.length();
for (int i = 0; i < length; i++) {
sb.append(KeyStr.charAt((int) Math.round(Math.random() * (len - 1))));
}
return sb.toString();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy