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에 의해 파괴되어 실제로 몇 년 동안 비밀을 유지했습니다. 일부 사람들은 여전히 3DES 가 안전 하다고 주장하지만 3DES의 약점을 발견하고 분석 한 연구 논문이 상당히 많습니다 .

암호화 모드

암호화는 블록 암호를 가져와 특정 체계를 사용하여 임의성을 키와 결합하여 키를 아는 한 뒤집을 수있는 무언가를 만들 때 만들어집니다. 이것을 암호화 모드라고합니다.

다음은 암호화 모드와 ECB로 알려진 가장 간단한 모드의 예입니다. 현재 상황을 시각적으로 이해할 수 있습니다.

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");

키와 비밀번호

또 다른 중요한 점은 암호화와 관련하여 키와 암호가 동일하지 않다는 것입니다. 암호화 키는 안전한 것으로 간주 되려면 일정량의 엔트로피와 임의성이 있어야합니다. 따라서 키를 생성하려면 적절한 암호화 라이브러리를 사용해야합니다.

그래서 당신은 실제로 두 가지 구현을 할 수 있습니다. 첫 번째는 RandomKey Generation위해이 StackOverflow 스레드에서 찾은 코드를 사용하는 입니다. 이 솔루션은 안전한 난수 생성기를 사용하여 처음부터 사용할 수있는 키를 만듭니다.

덜 안전한 다른 옵션은 암호와 같은 사용자 입력을 사용하는 것입니다. 우리가 논의한 문제는 암호에 엔트로피가 충분하지 않다는 것입니다. 따라서 암호를 취하고 강화하는 알고리즘 인 PBKDF2 를 사용해야 합니다. 여기 내가 좋아 하는 StackOverflow 구현이 있습니다. 그러나 Google Tink 라이브러리에는이 모든 것이 내장되어 있으므로 활용해야합니다.

안드로이드 개발자

여기서 지적해야 할 중요한 점은 안드로이드 코드가 리버스 엔지니어링 가능하며 대부분의 경우 자바 코드도 마찬가지라는 것입니다. 암호를 일반 텍스트로 코드에 저장하면 의미합니다. 해커가 쉽게 검색 할 수 있습니다. 일반적으로 이러한 유형의 암호화에는 비대칭 암호화 등을 사용하려고합니다. 이것은이 게시물의 범위를 벗어나므로 다이빙을 피할 것입니다.

2013 년흥미로운 기사 : 안드로이드에서 구현 한 Crypto 구현의 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();

}


}


답변

2019 년 12 월 12 일 업데이트

CBC와 같은 다른 모드와 달리 GCM 모드에서는 IV를 예측할 수 없습니다. 유일한 요구 사항은 IV가 주어진 키를 가진 각 호출에 대해 고유해야한다는 것입니다. 주어진 키에 대해 한 번 반복되면 보안이 손상 될 수 있습니다. 이를 달성하는 쉬운 방법은 아래에 표시된 것처럼 강력한 의사 난수 생성기에서 난수 IV를 사용하는 것입니다.

시퀀스 또는 타임 스탬프를 IV로 사용하는 것도 가능하지만 들리는 것처럼 사소한 것은 아닙니다. 예를 들어, 시스템이 지속적 저장소에서 IV로 이미 사용 된 시퀀스를 올바르게 추적하지 않으면 시스템을 재부팅 한 후 호출로 IV를 반복 할 수 있습니다. 마찬가지로, 완벽한 시계는 없습니다. 컴퓨터 시계 등 조정

또한 2 ^ 32 호출마다 키를 회전해야합니다. IV 요구 사항에 대한 자세한 내용은이 답변NIST 권장 사항을 참조하십시오 .


이것은 다음 사항을 고려하여 방금 Java 8로 작성한 암호화 및 암호 해독 코드입니다. 누군가가 이것을 유용하게 사용하기를 바랍니다.

  1. 암호화 알고리즘 : 256 비트 키를 가진 블록 암호 AES는 충분히 안전한 것으로 간주됩니다. 완전한 메시지를 암호화하려면 모드를 선택해야합니다. 기밀성과 무결성을 모두 제공하는 인증 된 암호화가 권장됩니다. GCM, CCM 및 EAX는 가장 일반적으로 사용되는 인증 된 암호화 모드입니다. GCM이 일반적으로 선호되며 GCM 전용 지침을 제공하는 인텔 아키텍처에서 성능이 우수합니다. 이 세 가지 모드는 모두 CTR 기반 (카운터 기반) 모드이므로 패딩이 필요하지 않습니다. 결과적으로 패딩 관련 공격에 취약하지 않습니다.

  2. GCM에는 초기화 벡터 (IV)가 필요합니다. IV는 비밀이 아닙니다. 유일한 요구 사항은 무작위이거나 예측할 수 없어야합니다. Java SecuredRandom에서이 클래스는 암호화 적으로 강력한 의사 난수를 생성합니다. 의사 난수 생성 알고리즘은 getInstance()메소드 에서 지정할 수 있습니다 . 그러나 Java 8 이후로 권장되는 방법은 getInstanceStrong()구성 및 제공하는 가장 강력한 알고리즘 을 사용하는 방법 을 사용 하는 것입니다.Provider

  3. NIST는 상호 ​​운용성, 효율성 및 단순 설계를 촉진하기 위해 GCM에 96 비트 IV를 권장합니다.

  4. 추가적인 보안을 보장하기 위해, 다음 구현 SecureRandom에서 의사 임의 바이트 생성의 2 ^ 16 바이트마다 생성 된 후에 다시 시드됩니다.

  5. 수신자는 IV를 알아야 암호 텍스트를 해독 할 수 있습니다. 따라서 IV는 암호문과 함께 전송되어야합니다. 일부 구현에서는 IV를 AD (Associated Data)로 전송합니다. 즉, 암호 텍스트와 IV 모두에서 인증 태그가 계산됩니다. 그러나 필수는 아닙니다. 의도적 인 공격이나 네트워크 / 파일 시스템 오류로 인해 전송 중에 IV가 변경되면 인증 태그 유효성 검사가 실패하기 때문에 IV에 암호 텍스트를 미리 붙일 수 있습니다.

  6. 문자열을 변경할 수 없으므로 문자열을 사용하여 일반 텍스트 메시지 나 키를 보관할 수 없습니다. 따라서 사용 후에는 문자열을 지울 수 없습니다. 이러한 정리되지 않은 문자열은 메모리에 남아 힙 덤프에 표시 될 수 있습니다. 같은 이유로, 이러한 암호화 또는 암호 해독 방법을 호출하는 클라이언트는 더 이상 필요하지 않은 메시지 또는 키를 보유한 모든 변수 또는 배열을 지워야합니다.

  7. 일반적인 권장 사항에 따라 코드에 제공자를 하드 코딩하지 않았습니다.

  8. 마지막으로 네트워크 또는 스토리지를 통한 전송을 위해서는 키 또는 암호 텍스트가 Base64 인코딩을 사용하여 인코딩되어야합니다. Base64에 대한 자세한 내용은 여기를 참조하십시오 . Java 8 접근 방식을 따라야합니다

바이트 배열은 다음을 사용하여 지울 수 있습니다.

Arrays.fill(clearTextMessageByteArray, Byte.MIN_VALUE);

그러나, 자바 (8)로, 취소하는 쉬운 방법이 없습니다 SecretKeyspecSecretKey이 두 인터페이스의 구현으로 방법을 구현하지 않는 것 destroy()인터페이스를 Destroyable. 다음 코드에서는 리플렉션 SecretKeySpecSecretKey사용 을 지우는 별도의 메서드가 작성되었습니다 .

아래 언급 된 두 가지 접근 방법 중 하나를 사용하여 키를 생성해야합니다.

키는 비밀번호와 같은 비밀이지만 사람이 사용하는 비밀번호와 달리 키는 암호화 알고리즘에 의해 사용되므로 위의 방법으로 만 생성해야합니다.

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먼저 바이트 배열로 변환하고 위에서 소개 한 encryptdecrypt함수를 호출해야합니다 . 다른 입력 키는 위에 제공된 코드 스 니펫을 사용하여 생성 할 수 있습니다.

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);


답변

이건 어때요:

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;
}

나를 위해 잘 작동하고 다소 컴팩트합니다.


답변

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 서버와 같이 요청할 수 있습니다. 보안 요구를 충족시킬 수 있습니다.