Java에서 문자열을 암호화하는 방법
내가 필요한 것은 2D 바코드 (PDF-417)에 표시되는 문자열을 암호화하여 누군가가 스캔 할 아이디어를 얻을 때 읽을 수있는 문자열을 암호화하는 것입니다.
다른 요구 사항 :
- 복잡해서는 안됩니다
- RSA, PKI 인프라, 키 쌍 등으로 구성되어서는 안됩니다.
스누핑하는 사람들을 제거하고 데이터를 얻는 데 관심이있는 다른 회사를 쉽게 해독 할 수있을 정도로 간단해야합니다. 그들은 우리를 부르고 표준을 말하거나 암호 해독에 사용할 수있는 간단한 키를 제공합니다.
아마도 이러한 회사는 다른 기술을 사용할 수 있으므로 특정 플랫폼이나 기술과 관련이없는 표준을 고수하는 것이 좋습니다.
당신은 무엇을 제안합니까? 하고 일부 Java 클래스 있습니까 encrypt()
및 decrypt()
높은 보안 표준을 달성 많은 합병증없이는?
이 페이지는 Google을 통해 표시되는 첫 번째 페이지이며 모든 구현의 보안 취약점으로 인해 위험에 빠질 수 있으므로 원래 게시물에서 7 년 이 지났으므로 다른 사람들의 암호화에 대한 정보를 추가하기 위해이 게시물을 게시하고 있습니다. 컴퓨터 공학 석사 학위 를 소지하고 암호학을 공부하고 배우는 데 많은 시간을 보냈으므로 인터넷을 더 안전한 곳으로 만들기 위해 2 센트를 던졌습니다.
또한 주어진 상황에서 많은 구현이 안전 할 수 있지만 왜 실수로 실수를 저지르는가? 특별한 이유가없는 한 사용 가능한 가장 강력한 도구를 사용하십시오. 전반적으로 나는 도서관을 사용하고 가능한 한 중요한 세부 사항을 피하는 것이 좋습니다.
업데이트 4/5/18 : 권장 라이브러리를 이해하기 쉽고 Jasypt 에서 Google의 새로운 라이브러리 Tink로 변경하기 위해 일부 부분을 다시 작성했습니다 . 기존 설정에서 Jasypt 를 완전히 제거하는 것이 좋습니다 .
머리말
아래에서는 안전한 대칭 암호화의 기본 사항을 간략하게 설명하고 사람들이 표준 Java 라이브러리를 사용하여 자체적으로 암호화를 구현할 때 온라인에서 흔히 볼 수있는 실수를 지적합니다. Google의 새 라이브러리로 실행되는 모든 세부 정보를 건너 뛰 려면 프로젝트로 가져 와서 모든 암호화에 AES-GCM 모드를 사용하면 안전합니다.
이제 자바에서 암호화하는 방법에 대한 핵심 세부 사항을 배우려면 :)
블록 암호
먼저 대칭 키 블록 암호를 선택해야합니다. 블록 암호는 의사 난수 생성에 사용되는 컴퓨터 기능 / 프로그램입니다. Pseudo-Randomness는 Quantum Computer 이외의 컴퓨터가 실제 임의성과 그 차이를 구분할 수없는 가짜 임의성입니다. 블록 암호는 암호화의 구성 요소와 비슷하며 다른 모드 나 체계와 함께 사용하면 암호화를 만들 수 있습니다.
이제 오늘 블록 암호 알고리즘을 사용할 수에 대해에 있는지 확인 NEVER , 반복한다 결코 사용하지 DES를 , 난 절대 사용하지 말 것 3DES를 . Snowden의 NSA 릴리스조차도 가능한 한 유사 의사 랜덤에 가까운 지 확인할 수있는 유일한 블록 암호는 AES 256 입니다. AES 128도 있지만 차이점은 AES 256은 256 비트 블록에서 작동하지만 AES 128은 128 블록에서 작동한다는 것입니다. 일부 약점이 발견되었지만 AES 128은 모두 안전한 것으로 간주되지만 256은 견고합니다.
재미있는 사실 DES 는 NSA가 처음 설립되어 실제로 몇 년 동안 비밀을 유지했을 때 DES 가 깨졌으며 일부 사람들은 여전히 3DES 가 안전 하다고 주장하지만 3DES의 약점을 발견하고 분석 한 연구 논문이 상당히 많습니다 .
암호화 모드
암호화는 블록 암호를 가져와 특정 체계를 사용하여 임의성을 키와 결합하여 키를 아는 한 뒤집을 수있는 무언가를 만들 때 만들어집니다. 이것을 암호화 모드라고합니다.
다음은 암호화 모드와 ECB로 알려진 가장 간단한 모드의 예입니다. 현재 상황을 시각적으로 이해할 수 있습니다.
온라인에서 가장 일반적으로 볼 수있는 암호화 모드는 다음과 같습니다.
ECB CTR, CBC, GCM
나열된 모드 이외의 다른 모드가 있으며 연구원은 기존 문제를 개선하기 위해 항상 새로운 모드를 향해 노력하고 있습니다.
이제 구현과 안전한 것으로 넘어 갑시다. ECB를 사용하지 마십시오 . 유명한 Linux 펭귄이 보여주는 것처럼 반복되는 데이터를 숨기는 것이 나쁩니다 .
Java로 구현할 때 다음 코드를 사용하면 ECB 모드가 기본적으로 설정됩니다.
Cipher cipher = Cipher.getInstance("AES");
...이 취약점은 취약점입니다! 불행히도 이것은 자습서와 예제에서 StackOverflow와 온라인에서 모두 볼 수 있습니다.
논체와 IV
ECB 모드에서 발견 된 문제에 대한 응답으로 IV라고도하는 nounces가 생성되었습니다. 아이디어는 새로운 임의의 변수를 생성하고 모든 암호화에 첨부하여 동일한 두 메시지를 암호화 할 때 다르게 표시되도록하는 것입니다. 이 뒤에 숨겨진 아름다움은 IV 또는 논 스가 공개 지식이라는 것입니다. 즉, 침입자는 이것에 액세스 할 수 있지만 키가 없으면 그 지식으로 아무것도 할 수 없습니다.
내가 볼 수있는 일반적인 문제는 사람들이 코드에서 동일한 고정 값과 같이 IV를 정적 값으로 설정한다는 것입니다. 그리고 여기 IV를 반복하는 순간 실제로 암호화의 전체 보안을 손상시키는 함정이 있습니다.
무작위 IV 생성
SecureRandom randomSecureRandom = SecureRandom.getInstance("SHA1PRNG");
byte[] iv = new byte[cipher.getBlockSize()];
randomSecureRandom.nextBytes(iv);
IvParameterSpec ivParams = new IvParameterSpec(iv);
참고 : SHA1이 손상되었지만 SHA256 을이 사용 사례에 올바르게 구현하는 방법을 찾을 수 없으므로 누군가 가이 문제를 해결하고 업데이트하려는 경우 훌륭합니다! 또한 SHA1 공격은 거대한 클러스터에서 크랙되기까지 몇 년이 걸릴 수 있으므로 여전히 비 전통적입니다. 자세한 내용은 여기를 확인하십시오.
CTR 구현
CTR 모드에는 패딩이 필요하지 않습니다.
Cipher cipher = Cipher.getInstance("AES/CTR/NoPadding");
CBC 구현
CBC 모드를 구현하기로 선택한 경우 다음과 같이 PKCS7Padding을 사용하십시오.
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS7Padding");
CBC 및 CTR 취약점 및 GCM을 사용해야하는 이유
CBC 및 CTR과 같은 다른 모드는 안전하지만 공격자가 암호화 된 데이터를 뒤집어 해독 할 때 값을 변경하는 문제가 발생합니다. 가상의 은행 메시지 "Sell 100"을 암호화한다고 가정 해 봅시다. 암호화 된 메시지는이 "eu23ng"처럼 보이고 공격자가 1 비트를 "eu53ng"로 변경하고 갑자기 메시지를 해독하면 "Sell 900"으로 표시됩니다.
이를 피하기 위해 대부분의 인터넷은 GCM을 사용하며 HTTPS를 볼 때마다 GCM을 사용하고있을 것입니다. GCM은 암호화 된 메시지에 해시로 서명하고이 서명을 사용하여 메시지가 변경되지 않았는지 확인합니다.
복잡성으로 인해 GCM을 구현하지 않을 것입니다. 실수로 IV를 반복하면 GCM의 경우 키를 손상시킬 수 있기 때문에 Google의 새로운 라이브러리 Tink를 사용하는 것이 좋습니다 . 이것이 궁극적 인 보안 결함입니다. 새로운 연구원들은 IV 반복 방지 암호화 모드를 향해 노력하고 있습니다. IV를 반복하더라도 키가 위험하지는 않지만 아직 주류가 아닙니다.
이제 GCM을 구현 하려면 멋진 GCM 구현에 대한 링크가 있습니다. 그러나 보안을 보장 할 수 없거나 보안이 제대로 구현되었지만 기본을 잃을 경우. 또한 GCM에서는 패딩이 없습니다.
Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");
키와 비밀번호
또 다른 중요한 점은 암호화와 관련하여 키와 암호가 동일하지 않다는 것입니다. 암호화 키는 안전한 것으로 간주 되려면 일정량의 엔트로피와 임의성이 있어야합니다. 따라서 키를 생성하려면 적절한 암호화 라이브러리를 사용해야합니다.
따라서 실제로 두 가지 구현을 할 수 있습니다. 첫 번째는 이 KeyOverflow 스레드에서 찾은 코드 를 랜덤 키 생성에 사용하는 것 입니다. 이 솔루션은 안전한 난수 생성기를 사용하여 처음부터 사용할 수있는 키를 만듭니다.
덜 안전한 다른 옵션은 암호와 같은 사용자 입력을 사용하는 것입니다. 우리가 논의한 문제는 암호에 엔트로피가 충분하지 않기 때문에 암호를 취하고 강화하는 알고리즘 인 PBKDF2 를 사용해야 한다는 것입니다. 여기 내가 좋아 하는 StackOverflow 구현이 있습니다. 그러나 Google Tink 라이브러리에는이 모든 것이 내장되어 있으므로 활용해야합니다.
안드로이드 개발자
여기서 지적해야 할 중요한 점은 안드로이드 코드가 리버스 엔지니어링 가능하며 대부분의 경우 자바 코드도 마찬가지라는 것입니다. 암호를 일반 텍스트로 코드에 저장하면 의미합니다. 해커가 쉽게 검색 할 수 있습니다. 일반적으로 이러한 유형의 암호화에는 비대칭 암호화 등을 사용하려고합니다. 이것은이 게시물의 범위를 벗어나므로 다이빙을 피할 것입니다.
2013 흥미로운 독서 : 안드로이드의 암호화 구현의 88 %가 잘못 수행하고 있다고 지적한다.
마지막 생각들
다시 한 번 암호화를 위해 Java 라이브러리를 직접 구현하지 말고 Google Tink를 사용 하는 것이 좋습니다. 모든 알고리즘을 올바르게 구현하는 데 실제로 많은 노력을 기울 였으므로 두통을 덜어 줄 것입니다. 그런 다음에도 Tink github에서 발생하는 문제, 여기 저기 취약점 팝업을 확인하십시오.
질문이나 의견이 있으시면 언제든지 의견을 보내주십시오! 보안은 항상 변하고 있습니다. 보안을 유지하려면 최선을 다해야합니다. :)
DES , 3DES 또는 AES 와 같이 널리 사용 가능한 표준 대칭 사이퍼를 사용하는 것이 좋습니다 . 이것이 가장 안전한 알고리즘은 아니지만 많은 구현이 있으며 바코드의 정보를 해독 해야하는 사람에게 열쇠를 주면됩니다. javax.crypto.Cipher 는 여기서 작업하고 싶습니다.
암호화 할 바이트가 있다고 가정합니다
byte[] input;
다음으로 키와 초기화 벡터 바이트 가 필요합니다.
byte[] keyBytes;
byte[] ivBytes;
이제 선택한 알고리즘의 암호를 초기화 할 수 있습니다.
// wrap key data in Key/IV specs to pass to cipher
SecretKeySpec key = new SecretKeySpec(keyBytes, "DES");
IvParameterSpec ivSpec = new IvParameterSpec(ivBytes);
// create the cipher with the algorithm you choose
// see javadoc for Cipher class for more info, e.g.
Cipher cipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
암호화는 다음과 같습니다.
cipher.init(Cipher.ENCRYPT_MODE, key, ivSpec);
byte[] encrypted= new byte[cipher.getOutputSize(input.length)];
int enc_len = cipher.update(input, 0, input.length, encrypted, 0);
enc_len += cipher.doFinal(encrypted, enc_len);
그리고 이와 같은 해독 :
cipher.init(Cipher.DECRYPT_MODE, key, ivSpec);
byte[] decrypted = new byte[cipher.getOutputSize(enc_len)];
int dec_len = cipher.update(encrypted, 0, enc_len, decrypted, 0);
dec_len += cipher.doFinal(decrypted, dec_len);
경고
이것을 일종의 보안 측정으로 사용하지 마십시오.
이 게시물의 암호화 메커니즘은 일회용 패드이므로 공격자가 2 개의 암호화 된 메시지를 사용하여 비밀 키를 쉽게 복구 할 수 있습니다. XOR 2 암호화 된 메시지와 키를 얻습니다. 간단합니다!
무사에 의해 지적
lib의 또 다른 JAR을 피하기 위해 Sun의 JRE에있는 Sun의 Base64Encoder / Decoder를 사용하고 있습니다. OpenJDK 또는 다른 JRE를 사용한다는 점에서 위험합니다. 그 외에도 Encoder / Decoder와 함께 Apache commons lib를 사용해야하는 또 다른 이유가 있습니까?
public class EncryptUtils {
public static final String DEFAULT_ENCODING = "UTF-8";
static BASE64Encoder enc = new BASE64Encoder();
static BASE64Decoder dec = new BASE64Decoder();
public static String base64encode(String text) {
try {
return enc.encode(text.getBytes(DEFAULT_ENCODING));
} catch (UnsupportedEncodingException e) {
return null;
}
}//base64encode
public static String base64decode(String text) {
try {
return new String(dec.decodeBuffer(text), DEFAULT_ENCODING);
} catch (IOException e) {
return null;
}
}//base64decode
public static void main(String[] args) {
String txt = "some text to be encrypted";
String key = "key phrase used for XOR-ing";
System.out.println(txt + " XOR-ed to: " + (txt = xorMessage(txt, key)));
String encoded = base64encode(txt);
System.out.println(" is encoded to: " + encoded + " and that is decoding to: " + (txt = base64decode(encoded)));
System.out.print("XOR-ing back to original: " + xorMessage(txt, key));
}
public static String xorMessage(String message, String key) {
try {
if (message == null || key == null) return null;
char[] keys = key.toCharArray();
char[] mesg = message.toCharArray();
int ml = mesg.length;
int kl = keys.length;
char[] newmsg = new char[ml];
for (int i = 0; i < ml; i++) {
newmsg[i] = (char)(mesg[i] ^ keys[i % kl]);
}//for i
return new String(newmsg);
} catch (Exception e) {
return null;
}
}//xorMessage
}//class
고맙습니다 ive 누군가가 사용자를 찾은 코드를 사용 하여이 클래스를 만들었습니다
객체 암호
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.security.InvalidAlgorithmParameterException;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.IllegalBlockSizeException;
import javax.crypto.NoSuchPaddingException;
import javax.crypto.ShortBufferException;
import javax.crypto.spec.DESKeySpec;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
public class ObjectCrypter {
private Cipher deCipher;
private Cipher enCipher;
private SecretKeySpec key;
private IvParameterSpec ivSpec;
public ObjectCrypter(byte[] keyBytes, byte[] ivBytes) {
// wrap key data in Key/IV specs to pass to cipher
ivSpec = new IvParameterSpec(ivBytes);
// create the cipher with the algorithm you choose
// see javadoc for Cipher class for more info, e.g.
try {
DESKeySpec dkey = new DESKeySpec(keyBytes);
key = new SecretKeySpec(dkey.getKey(), "DES");
deCipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
enCipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
} catch (NoSuchAlgorithmException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (NoSuchPaddingException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (InvalidKeyException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
public byte[] encrypt(Object obj) throws InvalidKeyException, InvalidAlgorithmParameterException, IOException, IllegalBlockSizeException, ShortBufferException, BadPaddingException {
byte[] input = convertToByteArray(obj);
enCipher.init(Cipher.ENCRYPT_MODE, key, ivSpec);
return enCipher.doFinal(input);
// cipher.init(Cipher.ENCRYPT_MODE, key, ivSpec);
// byte[] encypted = new byte[cipher.getOutputSize(input.length)];
// int enc_len = cipher.update(input, 0, input.length, encypted, 0);
// enc_len += cipher.doFinal(encypted, enc_len);
// return encypted;
}
public Object decrypt( byte[] encrypted) throws InvalidKeyException, InvalidAlgorithmParameterException, IllegalBlockSizeException, BadPaddingException, IOException, ClassNotFoundException {
deCipher.init(Cipher.DECRYPT_MODE, key, ivSpec);
return convertFromByteArray(deCipher.doFinal(encrypted));
}
private Object convertFromByteArray(byte[] byteObject) throws IOException,
ClassNotFoundException {
ByteArrayInputStream bais;
ObjectInputStream in;
bais = new ByteArrayInputStream(byteObject);
in = new ObjectInputStream(bais);
Object o = in.readObject();
in.close();
return o;
}
private byte[] convertToByteArray(Object complexObject) throws IOException {
ByteArrayOutputStream baos;
ObjectOutputStream out;
baos = new ByteArrayOutputStream();
out = new ObjectOutputStream(baos);
out.writeObject(complexObject);
out.close();
return baos.toByteArray();
}
}
Jasypt 를 사용할 수 있습니다
Jasypt를 사용하면 암호를 암호화하고 확인하는 것이 간단 할 수 있습니다.
StrongTextEncryptor textEncryptor = new StrongTextEncryptor();
textEncryptor.setPassword(myEncryptionPassword);
암호화 :
String myEncryptedText = textEncryptor.encrypt(myText);
복호화 :
String plainText = textEncryptor.decrypt(myEncryptedText);
그레들 :
compile group: 'org.jasypt', name: 'jasypt', version: '1.9.2'
풍모:
Jasypt는 간편한 단방향 (다이제스트) 및 양방향 암호화 기술을 제공합니다.
기본 Java VM뿐만 아니라 모든 JCE 공급자와 함께 사용할 수있는 개방형 API입니다. Jasypt는 Bouncy Castle과 같은 잘 알려진 공급자와 쉽게 사용할 수 있습니다. 더 알아보기.
사용자 비밀번호 보안 강화 더 알아보기.
이진 암호화 지원. Jasypt는 바이너리 (바이트 배열)의 다이제스트 및 암호화를 허용합니다. 필요할 때 개체 나 파일을 암호화합니다 (예 : 인터넷을 통해 전송).
숫자 암호화 지원. 텍스트 및 이진 외에도 숫자 값의 다이제스트 및 암호화를 허용합니다 (Hibernate 지속성을 암호화 할 때 BigInteger 및 BigDecimal, 다른 숫자 유형이 지원됨). 더 알아보기.
완전히 스레드 안전합니다.
멀티 프로세서 / 멀티 코어 시스템에서 고성능을 달성하기 위해 암호화 / 디 버스터 풀링을 지원합니다.
경량 ( "라이트") 버전의 라이브러리를 포함하여 모바일 플랫폼과 같은 크기 제한 환경에서 관리 효율성을 향상시킵니다.
암호화를 처음 사용하는 사용자에게는 구성이 필요없는 쉬운 암호화 도구와 고급 사용자를위한 고도로 구성 가능한 표준 암호화 도구를 모두 제공합니다.
암호화 된 방식으로 매핑 된 엔터티의 필드를 유지하기위한 Hibernate 3 및 4 선택적 통합. 필드의 암호화는 최대 절전 모드 매핑 파일에 정의되어 있으며 나머지 응용 프로그램에서는 투명하게 유지됩니다 (민감한 개인 데이터, 읽기 가능한 사용자가 많은 데이터베이스에 유용). 텍스트, 이진, 숫자, 부울, 날짜 암호화 ... 자세히 알아보기
Spring 2, Spring 3.0 및 Spring 3.1에 대한 특정 통합 기능을 통해 Spring 애플리케이션에 완벽하게 통합 할 수 있습니다. jasypt의 모든 다이제스트 및 암호화 기는 Spring에서 쉽게 사용할 수 있도록 설계되었습니다 (인스턴스가 주입되고 ...). 또한 스레드로부터 안전하기 때문에 Spring과 같은 싱글 톤 지향 환경에서 동기화 걱정없이 사용할 수 있습니다. 자세히 알아보기 : Spring 2, Spring 3.0, Spring 3.1.
스프링 보안 (이전의 Acegi Security)은 보안 암호화를위한 암호 암호화 및 일치 작업을 수행하고 안전한 암호 암호화 메커니즘을 사용하여 사용자 암호의 보안을 향상시키고보다 높은 수준의 구성 및 제어를 제공하기위한 선택적 통합입니다. 더 알아보기.
데이터베이스 비밀번호와 같은 민감한 정보를 포함하여 애플리케이션 구성 파일의 일부 또는 전부를 암호화하는 고급 기능을 제공합니다. 암호화 된 구성을 일반 스프링 기반 및 / 또는 최대 절전 모드 지원 응용 프로그램에 완벽하게 통합합니다. 더 알아보기.
사용하기 쉬운 CLI (명령 줄 인터페이스) 도구를 제공하여 개발자가 암호화 된 데이터를 초기화하고 유지 관리 작업 또는 스크립트에 암호화 / 복호화 / 다이제스트 작업을 포함시킬 수 있습니다. 더 알아보기.
보안 애플리케이션에서 URL을보다 강력하게 암호화하기 위해 Apache Wicket에 통합됩니다.
개발자가 자신의 데이터에 대한 실제 작업을 더 잘 이해할 수 있도록 포괄적 인 안내서 및 javadoc 문서.
강력한 문자 집합 지원. 원본 문자 집합에 관계없이 텍스트를 적절히 암호화하고 요약합니다. 인코딩이나 플랫폼 문제없이 일본어, 한국어, 아랍어 등의 언어를 완벽하게 지원합니다.
매우 높은 수준의 구성 기능 : 개발자는 "암호화 기"에 지시하는 것과 같은 트릭을 구현하여 암호에 사용할 암호를 원격 HTTPS 서버와 같이 요청할 수 있습니다. 보안 요구를 충족시킬 수 있습니다.
이건 어때요:
private static byte[] xor(final byte[] input, final byte[] secret) {
final byte[] output = new byte[input.length];
if (secret.length == 0) {
throw new IllegalArgumentException("empty security key");
}
int spos = 0;
for (int pos = 0; pos < input.length; ++pos) {
output[pos] = (byte) (input[pos] ^ secret[spos]);
++spos;
if (spos >= secret.length) {
spos = 0;
}
}
return output;
}
나를 위해 잘 작동하고 다소 컴팩트합니다.
다음은 meta64.com에서 Spring Singleton으로 구현 한 것입니다. 작동하는 각 호출에 대해 사이퍼 인스턴스를 작성하려는 경우 '동기화 된'호출을 제거 할 수 있지만 '암호'는 스레드로부터 안전하지 않습니다.
import java.security.Key;
import javax.crypto.Cipher;
import javax.crypto.spec.SecretKeySpec;
import javax.xml.bind.DatatypeConverter;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
@Component
@Scope("singleton")
public class Encryptor {
@Value("${aeskey}")
private String keyStr;
private Key aesKey = null;
private Cipher cipher = null;
synchronized private void init() throws Exception {
if (keyStr == null || keyStr.length() != 16) {
throw new Exception("bad aes key configured");
}
if (aesKey == null) {
aesKey = new SecretKeySpec(keyStr.getBytes(), "AES");
cipher = Cipher.getInstance("AES");
}
}
synchronized public String encrypt(String text) throws Exception {
init();
cipher.init(Cipher.ENCRYPT_MODE, aesKey);
return toHexString(cipher.doFinal(text.getBytes()));
}
synchronized public String decrypt(String text) throws Exception {
init();
cipher.init(Cipher.DECRYPT_MODE, aesKey);
return new String(cipher.doFinal(toByteArray(text)));
}
public static String toHexString(byte[] array) {
return DatatypeConverter.printHexBinary(array);
}
public static byte[] toByteArray(String s) {
return DatatypeConverter.parseHexBinary(s);
}
/*
* DO NOT DELETE
*
* Use this commented code if you don't like using DatatypeConverter dependency
*/
// public static String toHexStringOld(byte[] bytes) {
// StringBuilder sb = new StringBuilder();
// for (byte b : bytes) {
// sb.append(String.format("%02X", b));
// }
// return sb.toString();
// }
//
// public static byte[] toByteArrayOld(String s) {
// int len = s.length();
// byte[] data = new byte[len / 2];
// for (int i = 0; i < len; i += 2) {
// data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4) + Character.digit(s.charAt(i +
// 1), 16));
// }
// return data;
// }
}
이것은 다음 사항을 고려하여 방금 Java 8로 작성한 암호화 및 암호 해독 코드입니다. 누군가가 이것을 유용하게 사용하기를 바랍니다.
암호화 알고리즘 : 256 비트 키를 가진 블록 암호 AES는 충분히 안전한 것으로 간주됩니다. 완전한 메시지를 암호화하려면 모드를 선택해야합니다. 기밀성과 무결성을 모두 제공하는 인증 된 암호화가 권장됩니다. GCM, CCM 및 EAX는 가장 일반적으로 사용되는 인증 된 암호화 모드입니다. GCM이 일반적으로 선호되며 GCM에 대한 전용 지침을 제공하는 인텔 아키텍처에서 잘 수행됩니다. 이 세 가지 모드는 모두 CTR 기반 (카운터 기반) 모드이므로 패딩이 필요하지 않습니다. 결과적으로 패딩 관련 공격에 취약하지 않습니다.
GCM에는 초기화 벡터 (IV)가 필요합니다. IV는 비밀이 아닙니다. 유일한 요구 사항은 무작위이거나 예측할 수 없어야합니다. Java
SecuredRandom
에서이 클래스는 암호화 적으로 강력한 의사 난수를 생성합니다. 의사 난수 생성 알고리즘은getInstance()
메소드 에서 지정할 수 있습니다 . 그러나 Java 8 이후로 권장되는 방법은getInstanceStrong()
구성 및 제공하는 가장 강력한 알고리즘 을 사용하는 방법 을 사용 하는 것입니다.Provider
NIST는 상호 운용성, 효율성 및 단순 설계를 촉진하기 위해 GCM에 96 비트 IV를 권장합니다.
추가적인 보안을 보장하기 위해 다음의 구현
SecureRandom
에서는 2 ^ 16 바이트의 의사 난수 바이트 생성을 생성 한 후 다시 시드됩니다.수신자는 IV를 알아야 암호 텍스트를 해독 할 수 있습니다. 따라서 IV는 암호문과 함께 전송되어야합니다. 일부 구현에서는 IV를 AD (Associated Data)로 전송합니다. 즉, 암호 텍스트와 IV 모두에서 인증 태그가 계산됩니다. 그러나 필수는 아닙니다. 의도적 인 공격이나 네트워크 / 파일 시스템 오류로 인해 전송 중에 IV가 변경되면 인증 태그 유효성 검사가 실패하기 때문에 IV에 암호 텍스트를 미리 붙일 수 있습니다.
문자열을 변경할 수 없으므로 문자열을 사용하여 일반 텍스트 메시지 나 키를 보관할 수 없습니다. 따라서 사용 후에는 문자열을 지울 수 없습니다. 삭제되지 않은 이러한 문자열은 메모리에 남아 힙 덤프에 표시 될 수 있습니다. 같은 이유로, 이러한 암호화 또는 암호 해독 방법을 호출하는 클라이언트는 더 이상 필요하지 않은 메시지 또는 키를 보유한 모든 변수 또는 배열을 지워야합니다.
일반적인 권장 사항에 따라 코드에 제공자를 하드 코딩하지 않았습니다.
마지막으로 네트워크 또는 스토리지를 통한 전송을 위해서는 키 또는 암호 텍스트가 Base64 인코딩을 사용하여 인코딩되어야합니다. Base64에 대한 자세한 내용은 여기를 참조하십시오 . Java 8 접근 방식을 따라야합니다
바이트 배열은 다음을 사용하여 지울 수 있습니다.
Arrays.fill(clearTextMessageByteArray, Byte.MIN_VALUE);
그러나, 자바 (8)로, 취소하는 쉬운 방법이 없습니다 SecretKeyspec
및 SecretKey
이 두 인터페이스의 구현으로 방법을 구현하지 않는 것 destroy()
인터페이스를 Destroyable
. 다음 코드에서는 리플렉션 SecretKeySpec
및 SecretKey
사용 을 지우는 별도의 메서드가 작성되었습니다 .
아래 언급 된 두 가지 접근 방법 중 하나를 사용하여 키를 생성해야합니다.
키는 비밀번호와 같은 비밀이지만 사람이 사용하는 비밀번호와 달리 키는 암호화 알고리즘에 의해 사용되므로 위의 방법으로 만 생성해야합니다.
package com.sapbasu.javastudy;
import java.lang.reflect.Field;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import javax.crypto.Cipher;
import javax.crypto.spec.GCMParameterSpec;
import javax.crypto.spec.SecretKeySpec;
public class Crypto {
private static final int AUTH_TAG_SIZE = 128; // bits
// NIST recommendation: "For IVs, it is recommended that implementations
// restrict support to the length of 96 bits, to
// promote interoperability, efficiency, and simplicity of design."
private static final int IV_LEN = 12; // bytes
// number of random number bytes generated before re-seeding
private static final double PRNG_RESEED_INTERVAL = Math.pow(2, 16);
private static final String ENCRYPT_ALGO = "AES/GCM/NoPadding";
private static final List<Integer> ALLOWED_KEY_SIZES = Arrays
.asList(new Integer[] {128, 192, 256}); // bits
private static SecureRandom prng;
// Used to keep track of random number bytes generated by PRNG
// (for the purpose of re-seeding)
private static int bytesGenerated = 0;
public byte[] encrypt(byte[] input, SecretKeySpec key) throws Exception {
Objects.requireNonNull(input, "Input message cannot be null");
Objects.requireNonNull(key, "key cannot be null");
if (input.length == 0) {
throw new IllegalArgumentException("Length of message cannot be 0");
}
if (!ALLOWED_KEY_SIZES.contains(key.getEncoded().length * 8)) {
throw new IllegalArgumentException("Size of key must be 128, 192 or 256");
}
Cipher cipher = Cipher.getInstance(ENCRYPT_ALGO);
byte[] iv = getIV(IV_LEN);
GCMParameterSpec gcmParamSpec = new GCMParameterSpec(AUTH_TAG_SIZE, iv);
cipher.init(Cipher.ENCRYPT_MODE, key, gcmParamSpec);
byte[] messageCipher = cipher.doFinal(input);
// Prepend the IV with the message cipher
byte[] cipherText = new byte[messageCipher.length + IV_LEN];
System.arraycopy(iv, 0, cipherText, 0, IV_LEN);
System.arraycopy(messageCipher, 0, cipherText, IV_LEN,
messageCipher.length);
return cipherText;
}
public byte[] decrypt(byte[] input, SecretKeySpec key) throws Exception {
Objects.requireNonNull(input, "Input message cannot be null");
Objects.requireNonNull(key, "key cannot be null");
if (input.length == 0) {
throw new IllegalArgumentException("Input array cannot be empty");
}
byte[] iv = new byte[IV_LEN];
System.arraycopy(input, 0, iv, 0, IV_LEN);
byte[] messageCipher = new byte[input.length - IV_LEN];
System.arraycopy(input, IV_LEN, messageCipher, 0, input.length - IV_LEN);
GCMParameterSpec gcmParamSpec = new GCMParameterSpec(AUTH_TAG_SIZE, iv);
Cipher cipher = Cipher.getInstance(ENCRYPT_ALGO);
cipher.init(Cipher.DECRYPT_MODE, key, gcmParamSpec);
return cipher.doFinal(messageCipher);
}
public byte[] getIV(int bytesNum) {
if (bytesNum < 1) throw new IllegalArgumentException(
"Number of bytes must be greater than 0");
byte[] iv = new byte[bytesNum];
prng = Optional.ofNullable(prng).orElseGet(() -> {
try {
prng = SecureRandom.getInstanceStrong();
} catch (NoSuchAlgorithmException e) {
throw new RuntimeException("Wrong algorithm name", e);
}
return prng;
});
if (bytesGenerated > PRNG_RESEED_INTERVAL || bytesGenerated == 0) {
prng.setSeed(prng.generateSeed(bytesNum));
bytesGenerated = 0;
}
prng.nextBytes(iv);
bytesGenerated = bytesGenerated + bytesNum;
return iv;
}
private static void clearSecret(Destroyable key)
throws IllegalArgumentException, IllegalAccessException,
NoSuchFieldException, SecurityException {
Field keyField = key.getClass().getDeclaredField("key");
keyField.setAccessible(true);
byte[] encodedKey = (byte[]) keyField.get(key);
Arrays.fill(encodedKey, Byte.MIN_VALUE);
}
}
암호화 키는 주로 두 가지 방법으로 생성 할 수 있습니다.
비밀번호없이
KeyGenerator keyGen = KeyGenerator.getInstance("AES"); keyGen.init(KEY_LEN, SecureRandom.getInstanceStrong()); SecretKey secretKey = keyGen.generateKey(); SecretKeySpec secretKeySpec = new SecretKeySpec(secretKey.getEncoded(), "AES"); Crypto.clearSecret(secretKey); // After encryption or decryption with key Crypto.clearSecret(secretKeySpec);
비밀번호로
SecureRandom random = SecureRandom.getInstanceStrong(); byte[] salt = new byte[32]; random.nextBytes(salt); PBEKeySpec keySpec = new PBEKeySpec(password, salt, iterations, keyLength); SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA256"); SecretKey secretKey = keyFactory.generateSecret(keySpec); SecretKeySpec secretKeySpec = new SecretKeySpec(secretKey.getEncoded(), "AES"); Crypto.clearSecret(secretKey); // After encryption or decryption with key Crypto.clearSecret(secretKeySpec);
주석을 기반으로 업데이트
@MaartenBodewes가 지적한 것처럼 내 대답은 String
질문에 필요한대로 처리하지 못했습니다 . 따라서 누군가 가이 답변을 우연히 발견하고 처리에 대해 궁금해하는 경우를 대비하여 그 차이를 메우려 고합니다 String
.
답변의 앞부분에서 알 수 있듯이 a에서 민감한 정보를 처리하는 String
것은 일반적으로 String
불변이므로 사용 후에는 정보를 지울 수 없습니다. 그리고 우리가 알다시피, String
강력한 참조가 없더라도 가비지 수집기는 즉시 힙을 제거하기 위해 서두르지 않습니다. 따라서 String
프로그램에 액세스 할 수없는 경우에도 메모리에서 알 수없는 시간 동안 계속 유지됩니다. 그 문제는 해당 기간 동안 힙 덤프가 중요한 정보를 공개한다는 것입니다. 따라서 모든 민감한 정보를 바이트 배열 또는 char 배열로 처리 한 다음 목적이 달성되면 배열을 0으로 채우는 것이 항상 좋습니다.
그러나 모든 지식을 가지고도 암호화 할 민감한 정보가에있는 상황에서 여전히 String
바이트 배열로 변환하고 위에 소개 된 encrypt
및 decrypt
함수를 호출해야합니다 . 다른 입력 키는 위에 제공된 코드 스 니펫을 사용하여 생성 할 수 있습니다.
String
다음과 같은 방법으로 A 를 바이트로 변환 할 수 있습니다.
byte[] inputBytes = inputString.getBytes(StandardCharsets.UTF_8);
Java 8 String
부터는 UTF-16
인코딩 과 함께 내부적으로 힙에 저장됩니다 . 그러나 UTF-8
여기서는 UTF-16
특히 ASCII 문자의 경우보다 적은 공간을 차지하므로 여기에서 사용 했습니다 .
마찬가지로, 암호화 된 바이트 배열은 다음과 같이 문자열로 변환 될 수도 있습니다.
String encryptedString = new String(encryptedBytes, StandardCharsets.UTF_8);
https://www.bouncycastle.org/ 와 같은 것을 사용 하는 것이 좋습니다. 여러 개의 다른 암호로 원하는 것을 암호화 할 수있는 미리 빌드 된 라이브러리입니다. 스누핑으로부터 보호하려는 것만 알고 있지만 실제로는 Base64를 사용한다고해서 실제로 정보를 보호 할 수는 없습니다.
다음은 Java가 지원하는 내용을 읽을 수있는 링크입니다.
이 예는 AES, Blowfish, RC2, 3DES 등과 같은 대칭 암호화 알고리즘을 사용하여 많은 양의 데이터를 암호화하는 방법을 보여줍니다. 데이터는 EncryptBytes, EncryptString, EncryptBytesENC 또는 EncryptStringENC와 같은 암호화 방법 중 하나로 청크로 전달됩니다. (메소드 이름은 입력 유형 (문자열 또는 바이트 배열)과 반환 유형 (인코딩 된 문자열 또는 바이트 배열)을 나타냅니다. FirstChunk 및 LastChunk 속성은 청크가 스트림에서 첫 번째, 중간 또는 마지막인지 여부를 나타내는 데 사용됩니다. 기본적으로 FirstChunk와 LastChunk는 모두 true이며, 이는 전달 된 데이터가 전체 양임을 의미합니다.
여기에 단지와 간단한 솔루션 java.*
및 javax.crypto.*
제공 바이트의 암호화를위한 종속 기밀성 및 무결성을 . 킬로바이트 단위의 단문 메시지에 대해 선택된 일반 텍스트 공격 하에서 구별 할 수 없습니다 .
패딩이없는 모드 AES
에서 사용 되며 GCM
128 비트 키는 PBKDF2
많은 반복과 제공된 암호의 정적 소금으로 파생됩니다 . 이렇게하면 무차별 암호를 강제로 입력하고 전체 키에 엔트로피를 분산시킬 수 있습니다.
무작위 초기화 벡터 (IV)가 생성되며 암호문 앞에 추가됩니다. 또한 정적 바이트 0x01
는 '버전'으로 첫 번째 바이트로 추가됩니다.
전체 메시지는에서 생성 한 메시지 인증 코드 (MAC)로 들어갑니다 AES/GCM
.
기밀성 과 무결성을 제공하는 외부 종속성 암호화 클래스가 없습니다 .
package ch.n1b.tcrypt.utils;
import java.nio.charset.StandardCharsets;
import java.security.InvalidAlgorithmParameterException;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import java.security.NoSuchProviderException;
import java.security.SecureRandom;
import java.security.spec.InvalidKeySpecException;
import java.security.spec.KeySpec;
import javax.crypto.*;
import javax.crypto.spec.GCMParameterSpec;
import javax.crypto.spec.PBEKeySpec;
import javax.crypto.spec.SecretKeySpec;
/**
* This class implements AES-GCM symmetric key encryption with a PBKDF2 derived password.
* It provides confidentiality and integrity of the plaintext.
*
* @author Thomas Richner
* @created 2018-12-07
*/
public class AesGcmCryptor {
// https://crypto.stackexchange.com/questions/26783/ciphertext-and-tag-size-and-iv-transmission-with-aes-in-gcm-mode
private static final byte VERSION_BYTE = 0x01;
private static final int VERSION_BYTE_LENGTH = 1;
private static final int AES_KEY_BITS_LENGTH = 128;
// fixed AES-GCM constants
private static final String GCM_CRYPTO_NAME = "AES/GCM/NoPadding";
private static final int GCM_IV_BYTES_LENGTH = 12;
private static final int GCM_TAG_BYTES_LENGTH = 16;
// can be tweaked, more iterations = more compute intensive to brute-force password
private static final int PBKDF2_ITERATIONS = 1024;
// protects against rainbow tables
private static final byte[] PBKDF2_SALT = hexStringToByteArray("4d3fe0d71d2abd2828e7a3196ea450d4");
public String encryptString(char[] password, String plaintext) throws CryptoException {
byte[] encrypted = null;
try {
encrypted = encrypt(password, plaintext.getBytes(StandardCharsets.UTF_8));
} catch (NoSuchAlgorithmException | NoSuchPaddingException | InvalidKeyException //
| InvalidAlgorithmParameterException | IllegalBlockSizeException | BadPaddingException //
| InvalidKeySpecException e) {
throw new CryptoException(e);
}
return byteArrayToHexString(encrypted);
}
public String decryptString(char[] password, String ciphertext)
throws CryptoException {
byte[] ct = hexStringToByteArray(ciphertext);
byte[] plaintext = null;
try {
plaintext = decrypt(password, ct);
} catch (AEADBadTagException e) {
throw new CryptoException(e);
} catch ( //
NoSuchPaddingException | NoSuchAlgorithmException | InvalidKeySpecException //
| InvalidKeyException | InvalidAlgorithmParameterException | IllegalBlockSizeException //
| BadPaddingException e) {
throw new CryptoException(e);
}
return new String(plaintext, StandardCharsets.UTF_8);
}
/**
* Decrypts an AES-GCM encrypted ciphertext and is
* the reverse operation of {@link AesGcmCryptor#encrypt(char[], byte[])}
*
* @param password passphrase for decryption
* @param ciphertext encrypted bytes
* @return plaintext bytes
* @throws NoSuchPaddingException
* @throws NoSuchAlgorithmException
* @throws NoSuchProviderException
* @throws InvalidKeySpecException
* @throws InvalidAlgorithmParameterException
* @throws InvalidKeyException
* @throws BadPaddingException
* @throws IllegalBlockSizeException
* @throws IllegalArgumentException if the length or format of the ciphertext is bad
* @throws CryptoException
*/
public byte[] decrypt(char[] password, byte[] ciphertext)
throws NoSuchPaddingException, NoSuchAlgorithmException, InvalidKeySpecException,
InvalidAlgorithmParameterException, InvalidKeyException, BadPaddingException, IllegalBlockSizeException {
// input validation
if (ciphertext == null) {
throw new IllegalArgumentException("ciphertext cannot be null");
}
if (ciphertext.length <= VERSION_BYTE_LENGTH + GCM_IV_BYTES_LENGTH + GCM_TAG_BYTES_LENGTH) {
throw new IllegalArgumentException("ciphertext too short");
}
// the version must match, we don't decrypt other versions
if (ciphertext[0] != VERSION_BYTE) {
throw new IllegalArgumentException("wrong version: " + ciphertext[0]);
}
// input seems legit, lets decrypt and check integrity
// derive key from password
SecretKey key = deriveAesKey(password, PBKDF2_SALT, AES_KEY_BITS_LENGTH);
// init cipher
Cipher cipher = Cipher.getInstance(GCM_CRYPTO_NAME);
GCMParameterSpec params = new GCMParameterSpec(GCM_TAG_BYTES_LENGTH * 8,
ciphertext,
VERSION_BYTE_LENGTH,
GCM_IV_BYTES_LENGTH
);
cipher.init(Cipher.DECRYPT_MODE, key, params);
final int ciphertextOffset = VERSION_BYTE_LENGTH + GCM_IV_BYTES_LENGTH;
// add version and IV to MAC
cipher.updateAAD(ciphertext, 0, ciphertextOffset);
// decipher and check MAC
return cipher.doFinal(ciphertext, ciphertextOffset, ciphertext.length - ciphertextOffset);
}
/**
* Encrypts a plaintext with a password.
* <p>
* The encryption provides the following security properties:
* Confidentiality + Integrity
* <p>
* This is achieved my using the AES-GCM AEAD blockmode with a randomized IV.
* <p>
* The tag is calculated over the version byte, the IV as well as the ciphertext.
* <p>
* Finally the encrypted bytes have the following structure:
* <pre>
* +-------------------------------------------------------------------+
* | | | | |
* | version | IV bytes | ciphertext bytes | tag |
* | | | | |
* +-------------------------------------------------------------------+
* Length: 1B 12B len(plaintext) bytes 16B
* </pre>
* Note: There is no padding required for AES-GCM, but this also implies that
* the exact plaintext length is revealed.
*
* @param password password to use for encryption
* @param plaintext plaintext to encrypt
* @throws NoSuchAlgorithmException
* @throws NoSuchProviderException
* @throws NoSuchPaddingException
* @throws InvalidAlgorithmParameterException
* @throws InvalidKeyException
* @throws BadPaddingException
* @throws IllegalBlockSizeException
* @throws InvalidKeySpecException
*/
public byte[] encrypt(char[] password, byte[] plaintext)
throws NoSuchAlgorithmException, NoSuchPaddingException,
InvalidAlgorithmParameterException, InvalidKeyException, BadPaddingException, IllegalBlockSizeException,
InvalidKeySpecException {
// initialise random and generate IV (initialisation vector)
SecretKey key = deriveAesKey(password, PBKDF2_SALT, AES_KEY_BITS_LENGTH);
final byte[] iv = new byte[GCM_IV_BYTES_LENGTH];
SecureRandom random = SecureRandom.getInstanceStrong();
random.nextBytes(iv);
// encrypt
Cipher cipher = Cipher.getInstance(GCM_CRYPTO_NAME);
GCMParameterSpec spec = new GCMParameterSpec(GCM_TAG_BYTES_LENGTH * 8, iv);
cipher.init(Cipher.ENCRYPT_MODE, key, spec);
// add IV to MAC
final byte[] versionBytes = new byte[]{VERSION_BYTE};
cipher.updateAAD(versionBytes);
cipher.updateAAD(iv);
// encrypt and MAC plaintext
byte[] ciphertext = cipher.doFinal(plaintext);
// prepend VERSION and IV to ciphertext
byte[] encrypted = new byte[1 + GCM_IV_BYTES_LENGTH + ciphertext.length];
int pos = 0;
System.arraycopy(versionBytes, 0, encrypted, 0, VERSION_BYTE_LENGTH);
pos += VERSION_BYTE_LENGTH;
System.arraycopy(iv, 0, encrypted, pos, iv.length);
pos += iv.length;
System.arraycopy(ciphertext, 0, encrypted, pos, ciphertext.length);
return encrypted;
}
/**
* We derive a fixed length AES key with uniform entropy from a provided
* passphrase. This is done with PBKDF2/HMAC256 with a fixed count
* of iterations and a provided salt.
*
* @param password passphrase to derive key from
* @param salt salt for PBKDF2 if possible use a per-key salt, alternatively
* a random constant salt is better than no salt.
* @param keyLen number of key bits to output
* @return a SecretKey for AES derived from a passphrase
* @throws NoSuchAlgorithmException
* @throws InvalidKeySpecException
*/
private SecretKey deriveAesKey(char[] password, byte[] salt, int keyLen)
throws NoSuchAlgorithmException, InvalidKeySpecException {
if (password == null || salt == null || keyLen <= 0) {
throw new IllegalArgumentException();
}
SecretKeyFactory factory = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA256");
KeySpec spec = new PBEKeySpec(password, salt, PBKDF2_ITERATIONS, keyLen);
SecretKey pbeKey = factory.generateSecret(spec);
return new SecretKeySpec(pbeKey.getEncoded(), "AES");
}
/**
* Helper to convert hex strings to bytes.
* <p>
* May be used to read bytes from constants.
*/
private static byte[] hexStringToByteArray(String s) {
if (s == null) {
throw new IllegalArgumentException("Provided `null` string.");
}
int len = s.length();
if (len % 2 != 0) {
throw new IllegalArgumentException("Invalid length: " + len);
}
byte[] data = new byte[len / 2];
for (int i = 0; i < len - 1; i += 2) {
byte b = (byte) toHexDigit(s, i);
b <<= 4;
b |= toHexDigit(s, i + 1);
data[i / 2] = b;
}
return data;
}
private static int toHexDigit(String s, int pos) {
int d = Character.digit(s.charAt(pos), 16);
if (d < 0) {
throw new IllegalArgumentException("Cannot parse hex digit: " + s + " at " + pos);
}
return d;
}
private static String byteArrayToHexString(byte[] bytes) {
StringBuilder sb = new StringBuilder();
for (byte b : bytes) {
sb.append(String.format("%02X", b));
}
return sb.toString();
}
public class CryptoException extends Exception {
public CryptoException(Throwable cause) {
super(cause);
}
}
}
멋진 CLI를 사용하는 전체 프로젝트는 다음과 같습니다. https://github.com/trichner/tcrypt
편집 : 이제 적절 encryptString
하고decryptString
복사 / 붙여 넣기 솔루션은 다음과 같습니다. 또한 코드를 제공하지 않더라도 @ Konstantino의 답변 을 읽고 투표하는 것이 좋습니다 . 초기화 벡터 (IV)는 소금과 같습니다. 비밀로 유지 될 필요는 없습니다. 나는 GCM에 익숙하지 않으며 AAD는 선택 사항이며 특정 상황에서만 사용됩니다. 환경 변수에 키를 설정하십시오 SECRET_KEY_BASE
. KeePass 와 같은 것을 사용 하여 32 문자 암호를 생성하십시오. 이 솔루션은 내 Ruby 솔루션을 모델로합니다.
public static String encrypt(String s) {
try {
byte[] input = s.getBytes("UTF-8");
String keyString = System.getProperty("SECRET_KEY_BASE", System.getenv("SECRET_KEY_BASE"));
if (keyString == null || keyString.length() == 0) {
Logger.error(Utils.class, "encrypt()", "$SECRET_KEY_BASE is not set.");
return null;
}
byte[] keyBytes = keyString.getBytes("UTF-8");
SecretKeySpec keySpec = new SecretKeySpec(keyBytes, "AES");
// generate IV
SecureRandom secureRandom = SecureRandom.getInstanceStrong();
Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");
byte[] ivBytes = new byte[cipher.getBlockSize()];
secureRandom.nextBytes(ivBytes);
GCMParameterSpec gcmSpec = new GCMParameterSpec(96, ivBytes); // 96 bit tag length
cipher.init(Cipher.ENCRYPT_MODE, keySpec, gcmSpec);
// generate AAD
// byte[] aadBytes = new byte[cipher.getBlockSize()];
// secureRandom.nextBytes(aadBytes);
// cipher.updateAAD(aadBytes);
// encrypt
byte[] encrypted = cipher.doFinal(input);
byte[] returnBytes = new byte[ivBytes.length + encrypted.length];
// byte[] returnBytes = new byte[ivBytes.length + aadBytes.length + encrypted.length];
System.arraycopy(ivBytes, 0, returnBytes, 0, ivBytes.length);
// System.arraycopy(aadBytes, 0, returnBytes, ivBytes.length, aadBytes.length);
System.arraycopy(encrypted, 0, returnBytes, ivBytes.length, encrypted.length);
// System.arraycopy(encrypted, 0, returnBytes, ivBytes.length+aadBytes.length, encrypted.length);
String encryptedString = Base64.getEncoder().encodeToString(returnBytes);
return encryptedString;
} catch (UnsupportedEncodingException | NoSuchAlgorithmException | NoSuchPaddingException | InvalidKeyException |
InvalidAlgorithmParameterException | IllegalBlockSizeException | BadPaddingException e) {
Logger.error(Utils.class, "encrypt()", "Could not encrypt string: " + e.getMessage());
return null;
}
}
public static String decrypt(String s) {
if (s == null || s.length() == 0) return "";
try {
byte[] encrypted = Base64.getDecoder().decode(s);
String keyString = System.getProperty("SECRET_KEY_BASE", System.getenv("SECRET_KEY_BASE"));
if (keyString == null || keyString.length() == 0) {
Logger.error(Utils.class, "encrypt()", "$SECRET_KEY_BASE is not set.");
return null;
}
byte[] keyBytes = keyString.getBytes("UTF-8");
SecretKeySpec keySpec = new SecretKeySpec(keyBytes, "AES");
Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");
byte[] ivBytes = new byte[cipher.getBlockSize()];
System.arraycopy(encrypted, 0, ivBytes, 0, ivBytes.length);
GCMParameterSpec gcmSpec = new GCMParameterSpec(96, ivBytes);
cipher.init(Cipher.DECRYPT_MODE, keySpec, gcmSpec);
// cipher.updateAAD(encrypted, ivBytes.length, cipher.getBlockSize());
byte[] decrypted = cipher.doFinal(encrypted, cipher.getBlockSize(), encrypted.length - cipher.getBlockSize());
// byte[] decrypted = cipher.doFinal(encrypted, cipher.getBlockSize()*2, encrypted.length - cipher.getBlockSize()*2);
String decryptedString = new String(decrypted, "UTF-8");
return decryptedString;
} catch (NoSuchAlgorithmException | NoSuchPaddingException | UnsupportedEncodingException | InvalidKeyException |
InvalidAlgorithmParameterException | IllegalBlockSizeException | BadPaddingException e) {
Logger.error(Utils.class, "decrypt()", "Could not decrypt string: " + e.getMessage());
return null;
}
}
예를 들면 다음과 같습니다.
String s = "This is a test.";
String enc = Utils.encrypt(s);
System.out.println(enc);
// fQHfYjbD+xAuN5XzH2ojk/EWNeKXUrKRSfx8LU+5dpuKkM/pueCMBjKCZw==
String dec = Utils.decrypt(enc);
System.out.println(dec);
// This is a test.
많은 사람들이 이미 말했듯이 DES 또는 AES와 같이 과도하게 사용되는 표준 사이퍼를 사용해야합니다.
AES를 사용하여 Java에서 문자열을 암호화하고 해독하는 방법에 대한 간단한 예입니다 .
import org.apache.commons.codec.binary.Base64;
import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
public class EncryptorDemo {
public static String encrypt(String key, String randomVector, String value) {
try {
IvParameterSpec iv = new IvParameterSpec(randomVector.getBytes("UTF-8"));
SecretKeySpec skeySpec = new SecretKeySpec(key.getBytes("UTF-8"), "AES");
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5PADDING");
cipher.init(Cipher.ENCRYPT_MODE, skeySpec, iv);
byte[] encrypted = cipher.doFinal(value.getBytes());
System.out.println("encrypted text: " + Base64.encodeBase64String(encrypted));
return Base64.encodeBase64String(encrypted);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
public static String decrypt(String key, String randomVector, String encrypted) {
try {
IvParameterSpec iv = new IvParameterSpec(randomVector.getBytes("UTF-8"));
SecretKeySpec skeySpec = new SecretKeySpec(key.getBytes("UTF-8"), "AES");
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5PADDING");
cipher.init(Cipher.DECRYPT_MODE, skeySpec, iv);
byte[] originalText = cipher.doFinal(Base64.decodeBase64(encrypted));
System.out.println("decrypted text: " + new String(originalText));
return new String(originalText);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
public static void main(String[] args) {
String key = "JavasEncryptDemo"; // 128 bit key
String randomVector = "RandomJavaVector"; // 16 bytes IV
decrypt(key, randomVector, encrypt(key, randomVector, "Anything you want to encrypt!"));
}
}
예를 들어 암호화 / 암호 해독 코드 생성을 수행하는 자동화 된 도구를 고려할 수 있습니다. https://www.stringencrypt.com/java-encryption/
문자열 또는 파일 암호화에 대해 매번 다른 암호화 및 암호 해독 코드를 생성 할 수 있습니다.
RSA, AES 등을 사용하지 않고 빠른 문자열 암호화와 관련하여 매우 편리합니다.
샘플 결과 :
// encrypted with https://www.stringencrypt.com (v1.1.0) [Java]
// szTest = "Encryption in Java!"
String szTest = "\u9E3F\uA60F\uAE07\uB61B\uBE1F\uC62B\uCE2D\uD611" +
"\uDE03\uE5FF\uEEED\uF699\uFE3D\u071C\u0ED2\u1692" +
"\u1E06\u26AE\u2EDC";
for (int iatwS = 0, qUJQG = 0; iatwS < 19; iatwS++)
{
qUJQG = szTest.charAt(iatwS);
qUJQG ++;
qUJQG = ((qUJQG << 5) | ( (qUJQG & 0xFFFF) >> 11)) & 0xFFFF;
qUJQG -= iatwS;
qUJQG = (((qUJQG & 0xFFFF) >> 6) | (qUJQG << 10)) & 0xFFFF;
qUJQG ^= iatwS;
qUJQG -= iatwS;
qUJQG = (((qUJQG & 0xFFFF) >> 3) | (qUJQG << 13)) & 0xFFFF;
qUJQG ^= 0xFFFF;
qUJQG ^= 0xB6EC;
qUJQG = ((qUJQG << 8) | ( (qUJQG & 0xFFFF) >> 8)) & 0xFFFF;
qUJQG --;
qUJQG = (((qUJQG & 0xFFFF) >> 5) | (qUJQG << 11)) & 0xFFFF;
qUJQG ++;
qUJQG ^= 0xFFFF;
qUJQG += iatwS;
szTest = szTest.substring(0, iatwS) + (char)(qUJQG & 0xFFFF) + szTest.substring(iatwS + 1);
}
System.out.println(szTest);
우리는 항상 회사에서 사용합니다.
String s1="arshad";
char[] s2=s1.toCharArray();
int s3= s2.length;
System.out.println(s3);
int i=0;
// for(int j=0;j<s3;j++)
// System.out.println(s2[j]);
for(i=0;i<((s3)/2);i++) {
char z,f=10;
z=(char) (s2[i] * f);
s2[i]=s2[(s3-1)-i];
s2[(s3-1)-i]=z;
String b=new String(s2);
print(b); }
public static String encryptParams(String myTextInput) {
String myKey = "40674244454045cb9a70040a30e1c007";
String myVector = "@1B2c3D4e5F6g7H8";
String encData = "";
try{
JavaEncryprtionUtil encUtil = new JavaEncryprtionUtil();
encData = Base64.encodeToString(encUtil.encrypt(myTextInput.getBytes("UTF-8"), myKey.getBytes("UTF-8"), myVector.getBytes("UTF-8")),Base64.DEFAULT);
System.out.println(encData);
}catch(NoSuchAlgorithmException ex){
ex.printStackTrace();
}catch(NoSuchPaddingException ex){
ex.printStackTrace();
}catch(InvalidKeyException ex){
ex.printStackTrace();
}catch(InvalidAlgorithmParameterException ex){
ex.printStackTrace();
}catch(IllegalBlockSizeException ex){
ex.printStackTrace();
}catch(BadPaddingException ex){
ex.printStackTrace();
}catch(UnsupportedEncodingException ex){
ex.printStackTrace();
}
return encData;
}
참고 URL : https://stackoverflow.com/questions/1205135/how-to-encrypt-string-in-java
'development' 카테고리의 다른 글
문자열 변수를 사용하여 즉석에서 RegExps 만들기 (0) | 2020.07.01 |
---|---|
서비스를 시작한 후 Docker 컨테이너를 계속 실행하는 방법은 무엇입니까? (0) | 2020.07.01 |
기존 생산 프로젝트에 단위 테스트를 성공적으로 추가 할 수 있습니까? (0) | 2020.06.30 |
SDF 파일을 어떻게 열 수 있습니까 (SQL Server Compact Edition)? (0) | 2020.06.30 |
일련의 jQuery 연기로 어떻게 작업합니까? (0) | 2020.06.30 |