2018-01-24 19:17:44 -08:00
|
|
|
package org.thoughtcrime.securesms.crypto;
|
|
|
|
|
|
|
|
|
|
|
|
import android.support.annotation.NonNull;
|
2018-02-01 13:30:33 -08:00
|
|
|
import android.util.Pair;
|
2018-01-24 19:17:44 -08:00
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.FileOutputStream;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.OutputStream;
|
|
|
|
import java.security.InvalidAlgorithmParameterException;
|
|
|
|
import java.security.InvalidKeyException;
|
|
|
|
import java.security.NoSuchAlgorithmException;
|
2018-02-01 13:30:33 -08:00
|
|
|
import java.security.SecureRandom;
|
2018-01-24 19:17:44 -08:00
|
|
|
|
|
|
|
import javax.crypto.Cipher;
|
|
|
|
import javax.crypto.CipherOutputStream;
|
|
|
|
import javax.crypto.Mac;
|
|
|
|
import javax.crypto.NoSuchPaddingException;
|
|
|
|
import javax.crypto.spec.IvParameterSpec;
|
|
|
|
import javax.crypto.spec.SecretKeySpec;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructs an OutputStream that encrypts data written to it with the AttachmentSecret provided.
|
|
|
|
*
|
|
|
|
* The on-disk format is very simple, and intentionally no longer includes authentication.
|
|
|
|
*/
|
|
|
|
public class ModernEncryptingPartOutputStream {
|
|
|
|
|
2018-02-01 13:30:33 -08:00
|
|
|
public static Pair<byte[], OutputStream> createFor(@NonNull AttachmentSecret attachmentSecret, @NonNull File file, boolean inline)
|
2018-01-24 19:17:44 -08:00
|
|
|
throws IOException
|
|
|
|
{
|
2018-02-01 13:30:33 -08:00
|
|
|
byte[] random = new byte[32];
|
|
|
|
new SecureRandom().nextBytes(random);
|
|
|
|
|
2018-01-24 19:17:44 -08:00
|
|
|
try {
|
|
|
|
Mac mac = Mac.getInstance("HmacSHA256");
|
|
|
|
mac.init(new SecretKeySpec(attachmentSecret.getModernKey(), "HmacSHA256"));
|
|
|
|
|
|
|
|
FileOutputStream fileOutputStream = new FileOutputStream(file);
|
|
|
|
byte[] iv = new byte[16];
|
|
|
|
byte[] key = mac.doFinal(random);
|
|
|
|
|
|
|
|
Cipher cipher = Cipher.getInstance("AES/CTR/NoPadding");
|
|
|
|
cipher.init(Cipher.ENCRYPT_MODE, new SecretKeySpec(key, "AES"), new IvParameterSpec(iv));
|
|
|
|
|
2018-02-01 13:30:33 -08:00
|
|
|
if (inline) {
|
|
|
|
fileOutputStream.write(random);
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Pair<>(random, new CipherOutputStream(fileOutputStream, cipher));
|
2018-01-24 19:17:44 -08:00
|
|
|
} catch (NoSuchAlgorithmException | InvalidKeyException | InvalidAlgorithmParameterException | NoSuchPaddingException e) {
|
|
|
|
throw new AssertionError(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|