Moved stream crypto to crypto component.

This commit is contained in:
akwizgran
2014-12-29 19:55:05 +00:00
parent 02a485ace0
commit f316d64afa
33 changed files with 504 additions and 354 deletions

View File

@@ -0,0 +1,14 @@
package org.briarproject.api.crypto;
import java.io.IOException;
public interface StreamDecrypter {
/**
* Reads a frame, decrypts its payload into the given buffer and returns
* the payload length, or -1 if no more frames can be read from the stream.
* @throws java.io.IOException if an error occurs while reading the frame,
* or if authenticated decryption fails.
*/
int readFrame(byte[] payload) throws IOException;
}

View File

@@ -0,0 +1,18 @@
package org.briarproject.api.crypto;
import java.io.InputStream;
import org.briarproject.api.transport.StreamContext;
public interface StreamDecrypterFactory {
/** Creates a {@link StreamDecrypter} for decrypting a transport stream. */
StreamDecrypter createStreamDecrypter(InputStream in, int maxFrameLength,
StreamContext ctx);
/**
* Creates a {@link StreamDecrypter} for decrypting an invitation stream.
*/
StreamDecrypter createInvitationStreamDecrypter(InputStream in,
int maxFrameLength, byte[] secret, boolean alice);
}

View File

@@ -0,0 +1,13 @@
package org.briarproject.api.crypto;
import java.io.IOException;
public interface StreamEncrypter {
/** Encrypts the given frame and writes it to the stream. */
void writeFrame(byte[] payload, int payloadLength, boolean finalFrame)
throws IOException;
/** Flushes the stream. */
void flush() throws IOException;
}

View File

@@ -0,0 +1,18 @@
package org.briarproject.api.crypto;
import java.io.OutputStream;
import org.briarproject.api.transport.StreamContext;
public interface StreamEncrypterFactory {
/** Creates a {@link StreamEncrypter} for encrypting a transport stream. */
StreamEncrypter createStreamEncrypter(OutputStream out,
int maxFrameLength, StreamContext ctx);
/**
* Creates a {@link StreamEncrypter} for encrypting an invitation stream.
*/
StreamEncrypter createInvitationStreamEncrypter(OutputStream out,
int maxFrameLength, byte[] secret, boolean alice);
}

View File

@@ -1,13 +0,0 @@
package org.briarproject.api.transport;
import java.io.InputStream;
/** Decrypts and authenticates data received over an underlying transport. */
public interface StreamReader {
/**
* Returns an input stream from which the decrypted, authenticated data can
* be read.
*/
InputStream getInputStream();
}

View File

@@ -4,11 +4,17 @@ import java.io.InputStream;
public interface StreamReaderFactory { public interface StreamReaderFactory {
/** Creates a {@link StreamReader} for a transport connection. */ /**
StreamReader createStreamReader(InputStream in, int maxFrameLength, * Creates an {@link java.io.InputStream InputStream} for reading from a
* transport stream.
*/
InputStream createStreamReader(InputStream in, int maxFrameLength,
StreamContext ctx); StreamContext ctx);
/** Creates a {@link StreamReader} for an invitation connection. */ /**
StreamReader createInvitationStreamReader(InputStream in, * Creates an {@link java.io.InputStream InputStream} for reading from an
* invitation stream.
*/
InputStream createInvitationStreamReader(InputStream in,
int maxFrameLength, byte[] secret, boolean alice); int maxFrameLength, byte[] secret, boolean alice);
} }

View File

@@ -1,13 +0,0 @@
package org.briarproject.api.transport;
import java.io.OutputStream;
/** Encrypts and authenticates data to be sent over an underlying transport. */
public interface StreamWriter {
/**
* Returns an output stream to which unencrypted, unauthenticated data can
* be written.
*/
OutputStream getOutputStream();
}

View File

@@ -4,11 +4,17 @@ import java.io.OutputStream;
public interface StreamWriterFactory { public interface StreamWriterFactory {
/** Creates a {@link StreamWriter} for a transport connection. */ /**
StreamWriter createStreamWriter(OutputStream out, int maxFrameLength, * Creates an {@link java.io.OutputStream OutputStream} for writing to a
* transport stream
*/
OutputStream createStreamWriter(OutputStream out, int maxFrameLength,
StreamContext ctx); StreamContext ctx);
/** Creates a {@link StreamWriter} for an invitation connection. */ /**
StreamWriter createInvitationStreamWriter(OutputStream out, * Creates an {@link java.io.OutputStream OutputStream} for writing to an
* invitation stream.
*/
OutputStream createInvitationStreamWriter(OutputStream out,
int maxFrameLength, byte[] secret, boolean alice); int maxFrameLength, byte[] secret, boolean alice);
} }

View File

@@ -14,6 +14,8 @@ import javax.inject.Singleton;
import org.briarproject.api.crypto.CryptoComponent; import org.briarproject.api.crypto.CryptoComponent;
import org.briarproject.api.crypto.CryptoExecutor; import org.briarproject.api.crypto.CryptoExecutor;
import org.briarproject.api.crypto.PasswordStrengthEstimator; import org.briarproject.api.crypto.PasswordStrengthEstimator;
import org.briarproject.api.crypto.StreamDecrypterFactory;
import org.briarproject.api.crypto.StreamEncrypterFactory;
import org.briarproject.api.lifecycle.LifecycleManager; import org.briarproject.api.lifecycle.LifecycleManager;
import com.google.inject.AbstractModule; import com.google.inject.AbstractModule;
@@ -44,6 +46,8 @@ public class CryptoModule extends AbstractModule {
CryptoComponentImpl.class).in(Singleton.class); CryptoComponentImpl.class).in(Singleton.class);
bind(PasswordStrengthEstimator.class).to( bind(PasswordStrengthEstimator.class).to(
PasswordStrengthEstimatorImpl.class); PasswordStrengthEstimatorImpl.class);
bind(StreamDecrypterFactory.class).to(StreamDecrypterFactoryImpl.class);
bind(StreamEncrypterFactory.class).to(StreamEncrypterFactoryImpl.class);
} }
@Provides @Singleton @CryptoExecutor @Provides @Singleton @CryptoExecutor

View File

@@ -1,4 +1,4 @@
package org.briarproject.transport; package org.briarproject.crypto;
import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH; import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH;
import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH; import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;

View File

@@ -0,0 +1,42 @@
package org.briarproject.crypto;
import java.io.InputStream;
import javax.inject.Inject;
import org.briarproject.api.crypto.CryptoComponent;
import org.briarproject.api.crypto.SecretKey;
import org.briarproject.api.crypto.StreamDecrypter;
import org.briarproject.api.crypto.StreamDecrypterFactory;
import org.briarproject.api.transport.StreamContext;
class StreamDecrypterFactoryImpl implements StreamDecrypterFactory {
private final CryptoComponent crypto;
@Inject
StreamDecrypterFactoryImpl(CryptoComponent crypto) {
this.crypto = crypto;
}
public StreamDecrypter createStreamDecrypter(InputStream in,
int maxFrameLength, StreamContext ctx) {
byte[] secret = ctx.getSecret();
long streamNumber = ctx.getStreamNumber();
boolean alice = !ctx.getAlice();
// Derive the frame key
SecretKey frameKey = crypto.deriveFrameKey(secret, streamNumber, alice);
// Create the decrypter
return new StreamDecrypterImpl(in, crypto.getFrameCipher(), frameKey,
maxFrameLength);
}
public StreamDecrypter createInvitationStreamDecrypter(InputStream in,
int maxFrameLength, byte[] secret, boolean alice) {
// Derive the frame key
SecretKey frameKey = crypto.deriveFrameKey(secret, 0, alice);
// Create the decrypter
return new StreamDecrypterImpl(in, crypto.getFrameCipher(), frameKey,
maxFrameLength);
}
}

View File

@@ -1,4 +1,4 @@
package org.briarproject.transport; package org.briarproject.crypto;
import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH; import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH;
import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH; import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;
@@ -13,19 +13,20 @@ import java.security.GeneralSecurityException;
import org.briarproject.api.FormatException; import org.briarproject.api.FormatException;
import org.briarproject.api.crypto.AuthenticatedCipher; import org.briarproject.api.crypto.AuthenticatedCipher;
import org.briarproject.api.crypto.SecretKey; import org.briarproject.api.crypto.SecretKey;
import org.briarproject.api.crypto.StreamDecrypter;
class IncomingEncryptionLayer implements FrameReader { class StreamDecrypterImpl implements StreamDecrypter {
private final InputStream in; private final InputStream in;
private final AuthenticatedCipher frameCipher; private final AuthenticatedCipher frameCipher;
private final SecretKey frameKey; private final SecretKey frameKey;
private final byte[] iv, aad, ciphertext; private final byte[] iv, aad, plaintext, ciphertext;
private final int frameLength; private final int frameLength;
private long frameNumber; private long frameNumber;
private boolean finalFrame; private boolean finalFrame;
IncomingEncryptionLayer(InputStream in, AuthenticatedCipher frameCipher, StreamDecrypterImpl(InputStream in, AuthenticatedCipher frameCipher,
SecretKey frameKey, int frameLength) { SecretKey frameKey, int frameLength) {
this.in = in; this.in = in;
this.frameCipher = frameCipher; this.frameCipher = frameCipher;
@@ -33,12 +34,13 @@ class IncomingEncryptionLayer implements FrameReader {
this.frameLength = frameLength; this.frameLength = frameLength;
iv = new byte[IV_LENGTH]; iv = new byte[IV_LENGTH];
aad = new byte[AAD_LENGTH]; aad = new byte[AAD_LENGTH];
plaintext = new byte[frameLength - MAC_LENGTH];
ciphertext = new byte[frameLength]; ciphertext = new byte[frameLength];
frameNumber = 0; frameNumber = 0;
finalFrame = false; finalFrame = false;
} }
public int readFrame(byte[] frame) throws IOException { public int readFrame(byte[] payload) throws IOException {
if(finalFrame) return -1; if(finalFrame) return -1;
// Read the frame // Read the frame
int ciphertextLength = 0; int ciphertextLength = 0;
@@ -61,23 +63,25 @@ class IncomingEncryptionLayer implements FrameReader {
try { try {
frameCipher.init(false, frameKey, iv, aad); frameCipher.init(false, frameKey, iv, aad);
int decrypted = frameCipher.doFinal(ciphertext, 0, ciphertextLength, int decrypted = frameCipher.doFinal(ciphertext, 0, ciphertextLength,
frame, 0); plaintext, 0);
if(decrypted != plaintextLength) throw new RuntimeException(); if(decrypted != plaintextLength) throw new RuntimeException();
} catch(GeneralSecurityException e) { } catch(GeneralSecurityException e) {
throw new FormatException(); throw new FormatException();
} }
// Decode and validate the header // Decode and validate the header
finalFrame = FrameEncoder.isFinalFrame(frame); finalFrame = FrameEncoder.isFinalFrame(plaintext);
if(!finalFrame && ciphertextLength < frameLength) if(!finalFrame && ciphertextLength < frameLength)
throw new FormatException(); throw new FormatException();
int payloadLength = FrameEncoder.getPayloadLength(frame); int payloadLength = FrameEncoder.getPayloadLength(plaintext);
if(payloadLength > plaintextLength - HEADER_LENGTH) if(payloadLength > plaintextLength - HEADER_LENGTH)
throw new FormatException(); throw new FormatException();
// If there's any padding it must be all zeroes // If there's any padding it must be all zeroes
for(int i = HEADER_LENGTH + payloadLength; i < plaintextLength; i++) { for(int i = HEADER_LENGTH + payloadLength; i < plaintextLength; i++) {
if(frame[i] != 0) throw new FormatException(); if(plaintext[i] != 0) throw new FormatException();
} }
frameNumber++; frameNumber++;
// Copy the payload
System.arraycopy(plaintext, HEADER_LENGTH, payload, 0, payloadLength);
return payloadLength; return payloadLength;
} }
} }

View File

@@ -0,0 +1,49 @@
package org.briarproject.crypto;
import static org.briarproject.api.transport.TransportConstants.TAG_LENGTH;
import java.io.OutputStream;
import javax.inject.Inject;
import org.briarproject.api.crypto.CryptoComponent;
import org.briarproject.api.crypto.SecretKey;
import org.briarproject.api.crypto.StreamEncrypter;
import org.briarproject.api.crypto.StreamEncrypterFactory;
import org.briarproject.api.transport.StreamContext;
class StreamEncrypterFactoryImpl implements StreamEncrypterFactory {
private final CryptoComponent crypto;
@Inject
StreamEncrypterFactoryImpl(CryptoComponent crypto) {
this.crypto = crypto;
}
public StreamEncrypter createStreamEncrypter(OutputStream out,
int maxFrameLength, StreamContext ctx) {
byte[] secret = ctx.getSecret();
long streamNumber = ctx.getStreamNumber();
boolean alice = ctx.getAlice();
// Encode the tag
byte[] tag = new byte[TAG_LENGTH];
SecretKey tagKey = crypto.deriveTagKey(secret, alice);
crypto.encodeTag(tag, tagKey, streamNumber);
tagKey.erase();
// Derive the frame key
SecretKey frameKey = crypto.deriveFrameKey(secret, streamNumber, alice);
// Create the encrypter
return new StreamEncrypterImpl(out, crypto.getFrameCipher(), frameKey,
maxFrameLength, tag);
}
public StreamEncrypter createInvitationStreamEncrypter(OutputStream out,
int maxFrameLength, byte[] secret, boolean alice) {
// Derive the frame key
SecretKey frameKey = crypto.deriveFrameKey(secret, 0, alice);
// Create the encrypter
return new StreamEncrypterImpl(out, crypto.getFrameCipher(), frameKey,
maxFrameLength, null);
}
}

View File

@@ -1,4 +1,4 @@
package org.briarproject.transport; package org.briarproject.crypto;
import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH; import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH;
import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH; import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;
@@ -12,19 +12,20 @@ import java.security.GeneralSecurityException;
import org.briarproject.api.crypto.AuthenticatedCipher; import org.briarproject.api.crypto.AuthenticatedCipher;
import org.briarproject.api.crypto.SecretKey; import org.briarproject.api.crypto.SecretKey;
import org.briarproject.api.crypto.StreamEncrypter;
class OutgoingEncryptionLayer implements FrameWriter { class StreamEncrypterImpl implements StreamEncrypter {
private final OutputStream out; private final OutputStream out;
private final AuthenticatedCipher frameCipher; private final AuthenticatedCipher frameCipher;
private final SecretKey frameKey; private final SecretKey frameKey;
private final byte[] tag, iv, aad, ciphertext; private final byte[] tag, iv, aad, plaintext, ciphertext;
private final int frameLength; private final int frameLength;
private long frameNumber; private long frameNumber;
private boolean writeTag; private boolean writeTag;
OutgoingEncryptionLayer(OutputStream out, AuthenticatedCipher frameCipher, StreamEncrypterImpl(OutputStream out, AuthenticatedCipher frameCipher,
SecretKey frameKey, int frameLength, byte[] tag) { SecretKey frameKey, int frameLength, byte[] tag) {
this.out = out; this.out = out;
this.frameCipher = frameCipher; this.frameCipher = frameCipher;
@@ -33,13 +34,14 @@ class OutgoingEncryptionLayer implements FrameWriter {
this.tag = tag; this.tag = tag;
iv = new byte[IV_LENGTH]; iv = new byte[IV_LENGTH];
aad = new byte[AAD_LENGTH]; aad = new byte[AAD_LENGTH];
plaintext = new byte[frameLength - MAC_LENGTH];
ciphertext = new byte[frameLength]; ciphertext = new byte[frameLength];
frameNumber = 0; frameNumber = 0;
writeTag = (tag != null); writeTag = (tag != null);
} }
public void writeFrame(byte[] frame, int payloadLength, boolean finalFrame) public void writeFrame(byte[] payload, int payloadLength,
throws IOException { boolean finalFrame) throws IOException {
if(frameNumber > MAX_32_BIT_UNSIGNED) throw new IllegalStateException(); if(frameNumber > MAX_32_BIT_UNSIGNED) throw new IllegalStateException();
// Write the tag if required // Write the tag if required
if(writeTag) { if(writeTag) {
@@ -51,8 +53,6 @@ class OutgoingEncryptionLayer implements FrameWriter {
} }
writeTag = false; writeTag = false;
} }
// Encode the header
FrameEncoder.encodeHeader(frame, finalFrame, payloadLength);
// Don't pad the final frame // Don't pad the final frame
int plaintextLength, ciphertextLength; int plaintextLength, ciphertextLength;
if(finalFrame) { if(finalFrame) {
@@ -62,16 +62,19 @@ class OutgoingEncryptionLayer implements FrameWriter {
plaintextLength = frameLength - MAC_LENGTH; plaintextLength = frameLength - MAC_LENGTH;
ciphertextLength = frameLength; ciphertextLength = frameLength;
} }
// Encode the header
FrameEncoder.encodeHeader(plaintext, finalFrame, payloadLength);
// Copy the payload
System.arraycopy(payload, 0, plaintext, HEADER_LENGTH, payloadLength);
// If there's any padding it must all be zeroes // If there's any padding it must all be zeroes
for(int i = HEADER_LENGTH + payloadLength; i < plaintextLength; i++) { for(int i = HEADER_LENGTH + payloadLength; i < plaintextLength; i++)
frame[i] = 0; plaintext[i] = 0;
}
// Encrypt and authenticate the frame // Encrypt and authenticate the frame
FrameEncoder.encodeIv(iv, frameNumber); FrameEncoder.encodeIv(iv, frameNumber);
FrameEncoder.encodeAad(aad, frameNumber, plaintextLength); FrameEncoder.encodeAad(aad, frameNumber, plaintextLength);
try { try {
frameCipher.init(true, frameKey, iv, aad); frameCipher.init(true, frameKey, iv, aad);
int encrypted = frameCipher.doFinal(frame, 0, plaintextLength, int encrypted = frameCipher.doFinal(plaintext, 0, plaintextLength,
ciphertext, 0); ciphertext, 0);
if(encrypted != ciphertextLength) throw new RuntimeException(); if(encrypted != ciphertextLength) throw new RuntimeException();
} catch(GeneralSecurityException badCipher) { } catch(GeneralSecurityException badCipher) {

View File

@@ -29,9 +29,7 @@ import org.briarproject.api.serial.ReaderFactory;
import org.briarproject.api.serial.Writer; import org.briarproject.api.serial.Writer;
import org.briarproject.api.serial.WriterFactory; import org.briarproject.api.serial.WriterFactory;
import org.briarproject.api.system.Clock; import org.briarproject.api.system.Clock;
import org.briarproject.api.transport.StreamReader;
import org.briarproject.api.transport.StreamReaderFactory; import org.briarproject.api.transport.StreamReaderFactory;
import org.briarproject.api.transport.StreamWriter;
import org.briarproject.api.transport.StreamWriterFactory; import org.briarproject.api.transport.StreamWriterFactory;
/** A connection thread for the peer being Alice in the invitation protocol. */ /** A connection thread for the peer being Alice in the invitation protocol. */
@@ -51,9 +49,9 @@ class AliceConnector extends Connector {
Map<TransportId, TransportProperties> localProps, Map<TransportId, TransportProperties> localProps,
PseudoRandom random) { PseudoRandom random) {
super(crypto, db, readerFactory, writerFactory, streamReaderFactory, super(crypto, db, readerFactory, writerFactory, streamReaderFactory,
streamWriterFactory, authorFactory, groupFactory, streamWriterFactory, authorFactory, groupFactory, keyManager,
keyManager, connectionManager, clock, reuseConnection, group, connectionManager, clock, reuseConnection, group, plugin,
plugin, localAuthor, localProps, random); localAuthor, localProps, random);
} }
@Override @Override
@@ -131,14 +129,16 @@ class AliceConnector extends Connector {
if(LOG.isLoggable(INFO)) if(LOG.isLoggable(INFO))
LOG.info(pluginName + " confirmation succeeded"); LOG.info(pluginName + " confirmation succeeded");
int maxFrameLength = conn.getReader().getMaxFrameLength(); int maxFrameLength = conn.getReader().getMaxFrameLength();
StreamReader streamReader = // Create the readers
InputStream streamReader =
streamReaderFactory.createInvitationStreamReader(in, streamReaderFactory.createInvitationStreamReader(in,
maxFrameLength, secret, false); // Bob's stream maxFrameLength, secret, false); // Bob's stream
r = readerFactory.createReader(streamReader.getInputStream()); r = readerFactory.createReader(streamReader);
StreamWriter streamWriter = // Create the writers
OutputStream streamWriter =
streamWriterFactory.createInvitationStreamWriter(out, streamWriterFactory.createInvitationStreamWriter(out,
maxFrameLength, secret, true); // Alice's stream maxFrameLength, secret, true); // Alice's stream
w = writerFactory.createWriter(streamWriter.getOutputStream()); w = writerFactory.createWriter(streamWriter);
// Derive the invitation nonces // Derive the invitation nonces
byte[][] nonces = crypto.deriveInvitationNonces(secret); byte[][] nonces = crypto.deriveInvitationNonces(secret);
byte[] aliceNonce = nonces[0], bobNonce = nonces[1]; byte[] aliceNonce = nonces[0], bobNonce = nonces[1];

View File

@@ -29,9 +29,7 @@ import org.briarproject.api.serial.ReaderFactory;
import org.briarproject.api.serial.Writer; import org.briarproject.api.serial.Writer;
import org.briarproject.api.serial.WriterFactory; import org.briarproject.api.serial.WriterFactory;
import org.briarproject.api.system.Clock; import org.briarproject.api.system.Clock;
import org.briarproject.api.transport.StreamReader;
import org.briarproject.api.transport.StreamReaderFactory; import org.briarproject.api.transport.StreamReaderFactory;
import org.briarproject.api.transport.StreamWriter;
import org.briarproject.api.transport.StreamWriterFactory; import org.briarproject.api.transport.StreamWriterFactory;
/** A connection thread for the peer being Bob in the invitation protocol. */ /** A connection thread for the peer being Bob in the invitation protocol. */
@@ -51,9 +49,9 @@ class BobConnector extends Connector {
Map<TransportId, TransportProperties> localProps, Map<TransportId, TransportProperties> localProps,
PseudoRandom random) { PseudoRandom random) {
super(crypto, db, readerFactory, writerFactory, streamReaderFactory, super(crypto, db, readerFactory, writerFactory, streamReaderFactory,
streamWriterFactory, authorFactory, groupFactory, streamWriterFactory, authorFactory, groupFactory, keyManager,
keyManager, connectionManager, clock, reuseConnection, group, connectionManager, clock, reuseConnection, group, plugin,
plugin, localAuthor, localProps, random); localAuthor, localProps, random);
} }
@Override @Override
@@ -131,14 +129,16 @@ class BobConnector extends Connector {
if(LOG.isLoggable(INFO)) if(LOG.isLoggable(INFO))
LOG.info(pluginName + " confirmation succeeded"); LOG.info(pluginName + " confirmation succeeded");
int maxFrameLength = conn.getReader().getMaxFrameLength(); int maxFrameLength = conn.getReader().getMaxFrameLength();
StreamReader streamReader = // Create the readers
InputStream streamReader =
streamReaderFactory.createInvitationStreamReader(in, streamReaderFactory.createInvitationStreamReader(in,
maxFrameLength, secret, true); // Alice's stream maxFrameLength, secret, true); // Alice's stream
r = readerFactory.createReader(streamReader.getInputStream()); r = readerFactory.createReader(streamReader);
StreamWriter streamWriter = // Create the writers
OutputStream streamWriter =
streamWriterFactory.createInvitationStreamWriter(out, streamWriterFactory.createInvitationStreamWriter(out,
maxFrameLength, secret, false); // Bob's stream maxFrameLength, secret, false); // Bob's stream
w = writerFactory.createWriter(streamWriter.getOutputStream()); w = writerFactory.createWriter(streamWriter);
// Derive the nonces // Derive the nonces
byte[][] nonces = crypto.deriveInvitationNonces(secret); byte[][] nonces = crypto.deriveInvitationNonces(secret);
byte[] aliceNonce = nonces[0], bobNonce = nonces[1]; byte[] aliceNonce = nonces[0], bobNonce = nonces[1];

View File

@@ -6,6 +6,7 @@ import static org.briarproject.api.transport.TransportConstants.TAG_LENGTH;
import java.io.EOFException; import java.io.EOFException;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.io.OutputStream;
import java.util.concurrent.Executor; import java.util.concurrent.Executor;
import java.util.logging.Logger; import java.util.logging.Logger;
@@ -24,9 +25,7 @@ import org.briarproject.api.plugins.TransportConnectionReader;
import org.briarproject.api.plugins.TransportConnectionWriter; import org.briarproject.api.plugins.TransportConnectionWriter;
import org.briarproject.api.plugins.duplex.DuplexTransportConnection; import org.briarproject.api.plugins.duplex.DuplexTransportConnection;
import org.briarproject.api.transport.StreamContext; import org.briarproject.api.transport.StreamContext;
import org.briarproject.api.transport.StreamReader;
import org.briarproject.api.transport.StreamReaderFactory; import org.briarproject.api.transport.StreamReaderFactory;
import org.briarproject.api.transport.StreamWriter;
import org.briarproject.api.transport.StreamWriterFactory; import org.briarproject.api.transport.StreamWriterFactory;
import org.briarproject.api.transport.TagRecogniser; import org.briarproject.api.transport.TagRecogniser;
import org.briarproject.util.ByteUtils; import org.briarproject.util.ByteUtils;
@@ -97,11 +96,10 @@ class ConnectionManagerImpl implements ConnectionManager {
private MessagingSession createIncomingSession(StreamContext ctx, private MessagingSession createIncomingSession(StreamContext ctx,
TransportConnectionReader r) throws IOException { TransportConnectionReader r) throws IOException {
try { try {
StreamReader streamReader = streamReaderFactory.createStreamReader( InputStream streamReader = streamReaderFactory.createStreamReader(
r.getInputStream(), r.getMaxFrameLength(), ctx); r.getInputStream(), r.getMaxFrameLength(), ctx);
return messagingSessionFactory.createIncomingSession( return messagingSessionFactory.createIncomingSession(
ctx.getContactId(), ctx.getTransportId(), ctx.getContactId(), ctx.getTransportId(), streamReader);
streamReader.getInputStream());
} finally { } finally {
ByteUtils.erase(ctx.getSecret()); ByteUtils.erase(ctx.getSecret());
} }
@@ -110,11 +108,11 @@ class ConnectionManagerImpl implements ConnectionManager {
private MessagingSession createSimplexOutgoingSession(StreamContext ctx, private MessagingSession createSimplexOutgoingSession(StreamContext ctx,
TransportConnectionWriter w) throws IOException { TransportConnectionWriter w) throws IOException {
try { try {
StreamWriter streamWriter = streamWriterFactory.createStreamWriter( OutputStream streamWriter = streamWriterFactory.createStreamWriter(
w.getOutputStream(), w.getMaxFrameLength(), ctx); w.getOutputStream(), w.getMaxFrameLength(), ctx);
return messagingSessionFactory.createSimplexOutgoingSession( return messagingSessionFactory.createSimplexOutgoingSession(
ctx.getContactId(), ctx.getTransportId(), w.getMaxLatency(), ctx.getContactId(), ctx.getTransportId(), w.getMaxLatency(),
streamWriter.getOutputStream()); streamWriter);
} finally { } finally {
ByteUtils.erase(ctx.getSecret()); ByteUtils.erase(ctx.getSecret());
} }
@@ -123,11 +121,11 @@ class ConnectionManagerImpl implements ConnectionManager {
private MessagingSession createDuplexOutgoingSession(StreamContext ctx, private MessagingSession createDuplexOutgoingSession(StreamContext ctx,
TransportConnectionWriter w) throws IOException { TransportConnectionWriter w) throws IOException {
try { try {
StreamWriter streamWriter = streamWriterFactory.createStreamWriter( OutputStream streamWriter = streamWriterFactory.createStreamWriter(
w.getOutputStream(), w.getMaxFrameLength(), ctx); w.getOutputStream(), w.getMaxFrameLength(), ctx);
return messagingSessionFactory.createDuplexOutgoingSession( return messagingSessionFactory.createDuplexOutgoingSession(
ctx.getContactId(), ctx.getTransportId(), w.getMaxLatency(), ctx.getContactId(), ctx.getTransportId(), w.getMaxLatency(),
w.getMaxIdleTime(), streamWriter.getOutputStream()); w.getMaxIdleTime(), streamWriter);
} finally { } finally {
ByteUtils.erase(ctx.getSecret()); ByteUtils.erase(ctx.getSecret());
} }

View File

@@ -1,12 +0,0 @@
package org.briarproject.transport;
import java.io.IOException;
interface FrameReader {
/**
* Reads a frame into the given buffer and returns its payload length, or
* -1 if no more frames can be read from the connection.
*/
int readFrame(byte[] frame) throws IOException;
}

View File

@@ -1,13 +0,0 @@
package org.briarproject.transport;
import java.io.IOException;
interface FrameWriter {
/** Writes the given frame. */
void writeFrame(byte[] frame, int payloadLength, boolean finalFrame)
throws IOException;
/** Flushes the stream. */
void flush() throws IOException;
}

View File

@@ -4,37 +4,32 @@ import java.io.InputStream;
import javax.inject.Inject; import javax.inject.Inject;
import org.briarproject.api.crypto.CryptoComponent; import org.briarproject.api.crypto.StreamDecrypter;
import org.briarproject.api.crypto.SecretKey; import org.briarproject.api.crypto.StreamDecrypterFactory;
import org.briarproject.api.transport.StreamContext; import org.briarproject.api.transport.StreamContext;
import org.briarproject.api.transport.StreamReader;
import org.briarproject.api.transport.StreamReaderFactory; import org.briarproject.api.transport.StreamReaderFactory;
class StreamReaderFactoryImpl implements StreamReaderFactory { class StreamReaderFactoryImpl implements StreamReaderFactory {
private final CryptoComponent crypto; private final StreamDecrypterFactory streamDecrypterFactory;
@Inject @Inject
StreamReaderFactoryImpl(CryptoComponent crypto) { StreamReaderFactoryImpl(StreamDecrypterFactory streamDecrypterFactory) {
this.crypto = crypto; this.streamDecrypterFactory = streamDecrypterFactory;
} }
public StreamReader createStreamReader(InputStream in, public InputStream createStreamReader(InputStream in, int maxFrameLength,
int maxFrameLength, StreamContext ctx) { StreamContext ctx) {
byte[] secret = ctx.getSecret(); StreamDecrypter s = streamDecrypterFactory.createStreamDecrypter(in,
long streamNumber = ctx.getStreamNumber(); maxFrameLength, ctx);
boolean alice = !ctx.getAlice(); return new StreamReaderImpl(s, maxFrameLength);
SecretKey frameKey = crypto.deriveFrameKey(secret, streamNumber, alice);
FrameReader frameReader = new IncomingEncryptionLayer(in,
crypto.getFrameCipher(), frameKey, maxFrameLength);
return new StreamReaderImpl(frameReader, maxFrameLength);
} }
public StreamReader createInvitationStreamReader(InputStream in, public InputStream createInvitationStreamReader(InputStream in,
int maxFrameLength, byte[] secret, boolean alice) { int maxFrameLength, byte[] secret, boolean alice) {
SecretKey frameKey = crypto.deriveFrameKey(secret, 0, alice); StreamDecrypter s =
FrameReader frameReader = new IncomingEncryptionLayer(in, streamDecrypterFactory.createInvitationStreamDecrypter(in,
crypto.getFrameCipher(), frameKey, maxFrameLength); maxFrameLength, secret, alice);
return new StreamReaderImpl(frameReader, maxFrameLength); return new StreamReaderImpl(s, maxFrameLength);
} }
} }

View File

@@ -6,22 +6,18 @@ import static org.briarproject.api.transport.TransportConstants.MAC_LENGTH;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import org.briarproject.api.transport.StreamReader; import org.briarproject.api.crypto.StreamDecrypter;
class StreamReaderImpl extends InputStream implements StreamReader { class StreamReaderImpl extends InputStream {
private final FrameReader in; private final StreamDecrypter decrypter;
private final byte[] frame; private final byte[] payload;
private int offset = 0, length = 0; private int offset = 0, length = 0;
StreamReaderImpl(FrameReader in, int frameLength) { StreamReaderImpl(StreamDecrypter decrypter, int frameLength) {
this.in = in; this.decrypter = decrypter;
frame = new byte[frameLength - MAC_LENGTH]; payload = new byte[frameLength - HEADER_LENGTH - MAC_LENGTH];
}
public InputStream getInputStream() {
return this;
} }
@Override @Override
@@ -30,7 +26,7 @@ class StreamReaderImpl extends InputStream implements StreamReader {
if(length == -1) return -1; if(length == -1) return -1;
readFrame(); readFrame();
} }
int b = frame[offset] & 0xff; int b = payload[offset] & 0xff;
offset++; offset++;
length--; length--;
return b; return b;
@@ -48,7 +44,7 @@ class StreamReaderImpl extends InputStream implements StreamReader {
readFrame(); readFrame();
} }
len = Math.min(len, length); len = Math.min(len, length);
System.arraycopy(frame, offset, b, off, len); System.arraycopy(payload, offset, b, off, len);
offset += len; offset += len;
length -= len; length -= len;
return len; return len;
@@ -56,7 +52,7 @@ class StreamReaderImpl extends InputStream implements StreamReader {
private void readFrame() throws IOException { private void readFrame() throws IOException {
assert length == 0; assert length == 0;
offset = HEADER_LENGTH; offset = 0;
length = in.readFrame(frame); length = decrypter.readFrame(payload);
} }
} }

View File

@@ -1,46 +1,35 @@
package org.briarproject.transport; package org.briarproject.transport;
import static org.briarproject.api.transport.TransportConstants.TAG_LENGTH;
import java.io.OutputStream; import java.io.OutputStream;
import javax.inject.Inject; import javax.inject.Inject;
import org.briarproject.api.crypto.CryptoComponent; import org.briarproject.api.crypto.StreamEncrypter;
import org.briarproject.api.crypto.SecretKey; import org.briarproject.api.crypto.StreamEncrypterFactory;
import org.briarproject.api.transport.StreamContext; import org.briarproject.api.transport.StreamContext;
import org.briarproject.api.transport.StreamWriter;
import org.briarproject.api.transport.StreamWriterFactory; import org.briarproject.api.transport.StreamWriterFactory;
class StreamWriterFactoryImpl implements StreamWriterFactory { class StreamWriterFactoryImpl implements StreamWriterFactory {
private final CryptoComponent crypto; private final StreamEncrypterFactory streamEncrypterFactory;
@Inject @Inject
StreamWriterFactoryImpl(CryptoComponent crypto) { StreamWriterFactoryImpl(StreamEncrypterFactory streamEncrypterFactory) {
this.crypto = crypto; this.streamEncrypterFactory = streamEncrypterFactory;
} }
public StreamWriter createStreamWriter(OutputStream out, public OutputStream createStreamWriter(OutputStream out, int maxFrameLength,
int maxFrameLength, StreamContext ctx) { StreamContext ctx) {
byte[] secret = ctx.getSecret(); StreamEncrypter s = streamEncrypterFactory.createStreamEncrypter(out,
long streamNumber = ctx.getStreamNumber(); maxFrameLength, ctx);
boolean alice = ctx.getAlice(); return new StreamWriterImpl(s, maxFrameLength);
byte[] tag = new byte[TAG_LENGTH];
SecretKey tagKey = crypto.deriveTagKey(secret, alice);
crypto.encodeTag(tag, tagKey, streamNumber);
tagKey.erase();
SecretKey frameKey = crypto.deriveFrameKey(secret, streamNumber, alice);
FrameWriter frameWriter = new OutgoingEncryptionLayer(out,
crypto.getFrameCipher(), frameKey, maxFrameLength, tag);
return new StreamWriterImpl(frameWriter, maxFrameLength);
} }
public StreamWriter createInvitationStreamWriter(OutputStream out, public OutputStream createInvitationStreamWriter(OutputStream out,
int maxFrameLength, byte[] secret, boolean alice) { int maxFrameLength, byte[] secret, boolean alice) {
SecretKey frameKey = crypto.deriveFrameKey(secret, 0, alice); StreamEncrypter s =
FrameWriter frameWriter = new OutgoingEncryptionLayer(out, streamEncrypterFactory.createInvitationStreamEncrypter(out,
crypto.getFrameCipher(), frameKey, maxFrameLength, null); maxFrameLength, secret, alice);
return new StreamWriterImpl(frameWriter, maxFrameLength); return new StreamWriterImpl(s, maxFrameLength);
} }
} }

View File

@@ -6,7 +6,7 @@ import static org.briarproject.api.transport.TransportConstants.MAC_LENGTH;
import java.io.IOException; import java.io.IOException;
import java.io.OutputStream; import java.io.OutputStream;
import org.briarproject.api.transport.StreamWriter; import org.briarproject.api.crypto.StreamEncrypter;
/** /**
* A {@link org.briarproject.api.transport.StreamWriter StreamWriter} that * A {@link org.briarproject.api.transport.StreamWriter StreamWriter} that
@@ -15,43 +15,36 @@ import org.briarproject.api.transport.StreamWriter;
* <p> * <p>
* This class is not thread-safe. * This class is not thread-safe.
*/ */
class StreamWriterImpl extends OutputStream implements StreamWriter { class StreamWriterImpl extends OutputStream {
private final FrameWriter out; private final StreamEncrypter encrypter;
private final byte[] frame; private final byte[] payload;
private final int frameLength;
private int length = 0; private int length = 0;
StreamWriterImpl(FrameWriter out, int frameLength) { StreamWriterImpl(StreamEncrypter encrypter, int maxFrameLength) {
this.out = out; this.encrypter = encrypter;
this.frameLength = frameLength; payload = new byte[maxFrameLength - HEADER_LENGTH - MAC_LENGTH];
frame = new byte[frameLength - MAC_LENGTH];
}
public OutputStream getOutputStream() {
return this;
} }
@Override @Override
public void close() throws IOException { public void close() throws IOException {
writeFrame(true); writeFrame(true);
out.flush(); encrypter.flush();
super.close(); super.close();
} }
@Override @Override
public void flush() throws IOException { public void flush() throws IOException {
writeFrame(false); writeFrame(false);
out.flush(); encrypter.flush();
} }
@Override @Override
public void write(int b) throws IOException { public void write(int b) throws IOException {
frame[HEADER_LENGTH + length] = (byte) b; payload[length] = (byte) b;
length++; length++;
if(HEADER_LENGTH + length + MAC_LENGTH == frameLength) if(length == payload.length) writeFrame(false);
writeFrame(false);
} }
@Override @Override
@@ -61,21 +54,21 @@ class StreamWriterImpl extends OutputStream implements StreamWriter {
@Override @Override
public void write(byte[] b, int off, int len) throws IOException { public void write(byte[] b, int off, int len) throws IOException {
int available = frameLength - HEADER_LENGTH - length - MAC_LENGTH; int available = payload.length - length;
while(available <= len) { while(available <= len) {
System.arraycopy(b, off, frame, HEADER_LENGTH + length, available); System.arraycopy(b, off, payload, length, available);
length += available; length += available;
writeFrame(false); writeFrame(false);
off += available; off += available;
len -= available; len -= available;
available = frameLength - HEADER_LENGTH - length - MAC_LENGTH; available = payload.length - length;
} }
System.arraycopy(b, off, frame, HEADER_LENGTH + length, len); System.arraycopy(b, off, payload, length, len);
length += len; length += len;
} }
private void writeFrame(boolean finalFrame) throws IOException { private void writeFrame(boolean finalFrame) throws IOException {
out.writeFrame(frame, length, finalFrame); encrypter.writeFrame(payload, length, finalFrame);
length = 0; length = 0;
} }
} }

View File

@@ -102,6 +102,8 @@
<test name="org.briarproject.crypto.PasswordBasedKdfTest"/> <test name="org.briarproject.crypto.PasswordBasedKdfTest"/>
<test name="org.briarproject.crypto.PasswordStrengthEstimatorTest"/> <test name="org.briarproject.crypto.PasswordStrengthEstimatorTest"/>
<test name='org.briarproject.crypto.SecretKeyImplTest'/> <test name='org.briarproject.crypto.SecretKeyImplTest'/>
<test name='org.briarproject.crypto.StreamDecrypterImplTest'/>
<test name='org.briarproject.crypto.StreamEncrypterImplTest'/>
<test name='org.briarproject.db.BasicH2Test'/> <test name='org.briarproject.db.BasicH2Test'/>
<test name='org.briarproject.db.DatabaseCleanerImplTest'/> <test name='org.briarproject.db.DatabaseCleanerImplTest'/>
<test name='org.briarproject.db.DatabaseComponentImplTest'/> <test name='org.briarproject.db.DatabaseComponentImplTest'/>
@@ -126,10 +128,8 @@
<test name='org.briarproject.serial.ReaderImplTest'/> <test name='org.briarproject.serial.ReaderImplTest'/>
<test name='org.briarproject.serial.WriterImplTest'/> <test name='org.briarproject.serial.WriterImplTest'/>
<test name='org.briarproject.system.LinuxSeedProviderTest'/> <test name='org.briarproject.system.LinuxSeedProviderTest'/>
<test name='org.briarproject.transport.IncomingEncryptionLayerTest'/>
<test name='org.briarproject.transport.KeyManagerImplTest'/> <test name='org.briarproject.transport.KeyManagerImplTest'/>
<test name='org.briarproject.transport.KeyRotationIntegrationTest'/> <test name='org.briarproject.transport.KeyRotationIntegrationTest'/>
<test name='org.briarproject.transport.OutgoingEncryptionLayerTest'/>
<test name='org.briarproject.transport.ReorderingWindowTest'/> <test name='org.briarproject.transport.ReorderingWindowTest'/>
<test name='org.briarproject.transport.StreamReaderImplTest'/> <test name='org.briarproject.transport.StreamReaderImplTest'/>
<test name='org.briarproject.transport.StreamWriterImplTest'/> <test name='org.briarproject.transport.StreamWriterImplTest'/>

View File

@@ -7,6 +7,7 @@ import static org.junit.Assert.assertArrayEquals;
import java.io.ByteArrayInputStream; import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream; import java.io.ByteArrayOutputStream;
import java.io.InputStream; import java.io.InputStream;
import java.io.OutputStream;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
@@ -36,9 +37,7 @@ import org.briarproject.api.messaging.SubscriptionUpdate;
import org.briarproject.api.messaging.TransportUpdate; import org.briarproject.api.messaging.TransportUpdate;
import org.briarproject.api.messaging.UnverifiedMessage; import org.briarproject.api.messaging.UnverifiedMessage;
import org.briarproject.api.transport.StreamContext; import org.briarproject.api.transport.StreamContext;
import org.briarproject.api.transport.StreamReader;
import org.briarproject.api.transport.StreamReaderFactory; import org.briarproject.api.transport.StreamReaderFactory;
import org.briarproject.api.transport.StreamWriter;
import org.briarproject.api.transport.StreamWriterFactory; import org.briarproject.api.transport.StreamWriterFactory;
import org.briarproject.crypto.CryptoModule; import org.briarproject.crypto.CryptoModule;
import org.briarproject.db.DatabaseModule; import org.briarproject.db.DatabaseModule;
@@ -119,10 +118,10 @@ public class ProtocolIntegrationTest extends BriarTestCase {
ByteArrayOutputStream out = new ByteArrayOutputStream(); ByteArrayOutputStream out = new ByteArrayOutputStream();
StreamContext ctx = new StreamContext(contactId, transportId, StreamContext ctx = new StreamContext(contactId, transportId,
secret.clone(), 0, true); secret.clone(), 0, true);
StreamWriter streamWriter = streamWriterFactory.createStreamWriter(out, OutputStream streamWriter = streamWriterFactory.createStreamWriter(out,
MAX_FRAME_LENGTH, ctx); MAX_FRAME_LENGTH, ctx);
PacketWriter packetWriter = packetWriterFactory.createPacketWriter( PacketWriter packetWriter = packetWriterFactory.createPacketWriter(
streamWriter.getOutputStream()); streamWriter);
packetWriter.writeAck(new Ack(messageIds)); packetWriter.writeAck(new Ack(messageIds));
@@ -140,7 +139,7 @@ public class ProtocolIntegrationTest extends BriarTestCase {
transportProperties, 1); transportProperties, 1);
packetWriter.writeTransportUpdate(tu); packetWriter.writeTransportUpdate(tu);
streamWriter.getOutputStream().flush(); streamWriter.flush();
return out.toByteArray(); return out.toByteArray();
} }
@@ -151,10 +150,10 @@ public class ProtocolIntegrationTest extends BriarTestCase {
// FIXME: Check that the expected tag was received // FIXME: Check that the expected tag was received
StreamContext ctx = new StreamContext(contactId, transportId, StreamContext ctx = new StreamContext(contactId, transportId,
secret.clone(), 0, false); secret.clone(), 0, false);
StreamReader streamReader = streamReaderFactory.createStreamReader(in, InputStream streamReader = streamReaderFactory.createStreamReader(in,
MAX_FRAME_LENGTH, ctx); MAX_FRAME_LENGTH, ctx);
PacketReader packetReader = packetReaderFactory.createPacketReader( PacketReader packetReader = packetReaderFactory.createPacketReader(
streamReader.getInputStream()); streamReader);
// Read the ack // Read the ack
assertTrue(packetReader.hasAck()); assertTrue(packetReader.hasAck());

View File

@@ -1,4 +1,4 @@
package org.briarproject.transport; package org.briarproject.crypto;
import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH; import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH;
import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH; import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;
@@ -14,13 +14,12 @@ import org.briarproject.api.FormatException;
import org.briarproject.api.crypto.AuthenticatedCipher; import org.briarproject.api.crypto.AuthenticatedCipher;
import org.briarproject.api.crypto.CryptoComponent; import org.briarproject.api.crypto.CryptoComponent;
import org.briarproject.api.crypto.SecretKey; import org.briarproject.api.crypto.SecretKey;
import org.briarproject.crypto.CryptoModule;
import org.junit.Test; import org.junit.Test;
import com.google.inject.Guice; import com.google.inject.Guice;
import com.google.inject.Injector; import com.google.inject.Injector;
public class IncomingEncryptionLayerTest extends BriarTestCase { public class StreamDecrypterImplTest extends BriarTestCase {
// FIXME: This is an integration test, not a unit test // FIXME: This is an integration test, not a unit test
@@ -32,7 +31,7 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
private final AuthenticatedCipher frameCipher; private final AuthenticatedCipher frameCipher;
private final SecretKey frameKey; private final SecretKey frameKey;
public IncomingEncryptionLayerTest() { public StreamDecrypterImplTest() {
Injector i = Guice.createInjector(new CryptoModule(), Injector i = Guice.createInjector(new CryptoModule(),
new TestLifecycleModule(), new TestSystemModule()); new TestLifecycleModule(), new TestSystemModule());
crypto = i.getInstance(CryptoComponent.class); crypto = i.getInstance(CryptoComponent.class);
@@ -51,11 +50,11 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
System.arraycopy(frame1, 0, valid, FRAME_LENGTH, FRAME_LENGTH); System.arraycopy(frame1, 0, valid, FRAME_LENGTH, FRAME_LENGTH);
// Read the frames // Read the frames
ByteArrayInputStream in = new ByteArrayInputStream(valid); ByteArrayInputStream in = new ByteArrayInputStream(valid);
IncomingEncryptionLayer i = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypterImpl i = new StreamDecrypterImpl(in, frameCipher,
frameKey, FRAME_LENGTH); frameKey, FRAME_LENGTH);
byte[] buf = new byte[FRAME_LENGTH - MAC_LENGTH]; byte[] payload = new byte[MAX_PAYLOAD_LENGTH];
assertEquals(123, i.readFrame(buf)); assertEquals(123, i.readFrame(payload));
assertEquals(123, i.readFrame(buf)); assertEquals(123, i.readFrame(payload));
} }
@Test @Test
@@ -67,10 +66,10 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
System.arraycopy(frame, 0, truncated, 0, FRAME_LENGTH - 1); System.arraycopy(frame, 0, truncated, 0, FRAME_LENGTH - 1);
// Try to read the frame, which should fail due to truncation // Try to read the frame, which should fail due to truncation
ByteArrayInputStream in = new ByteArrayInputStream(truncated); ByteArrayInputStream in = new ByteArrayInputStream(truncated);
IncomingEncryptionLayer i = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypterImpl i = new StreamDecrypterImpl(in, frameCipher,
frameKey, FRAME_LENGTH); frameKey, FRAME_LENGTH);
try { try {
i.readFrame(new byte[FRAME_LENGTH - MAC_LENGTH]); i.readFrame(new byte[MAX_PAYLOAD_LENGTH]);
fail(); fail();
} catch(FormatException expected) {} } catch(FormatException expected) {}
} }
@@ -83,10 +82,10 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
frame[(int) (Math.random() * FRAME_LENGTH)] ^= 1; frame[(int) (Math.random() * FRAME_LENGTH)] ^= 1;
// Try to read the frame, which should fail due to modification // Try to read the frame, which should fail due to modification
ByteArrayInputStream in = new ByteArrayInputStream(frame); ByteArrayInputStream in = new ByteArrayInputStream(frame);
IncomingEncryptionLayer i = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypterImpl i = new StreamDecrypterImpl(in, frameCipher,
frameKey, FRAME_LENGTH); frameKey, FRAME_LENGTH);
try { try {
i.readFrame(new byte[FRAME_LENGTH - MAC_LENGTH]); i.readFrame(new byte[MAX_PAYLOAD_LENGTH]);
fail(); fail();
} catch(FormatException expected) {} } catch(FormatException expected) {}
} }
@@ -97,10 +96,10 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
byte[] frame = generateFrame(0, FRAME_LENGTH - 1, 123, false, false); byte[] frame = generateFrame(0, FRAME_LENGTH - 1, 123, false, false);
// Try to read the frame, which should fail due to invalid length // Try to read the frame, which should fail due to invalid length
ByteArrayInputStream in = new ByteArrayInputStream(frame); ByteArrayInputStream in = new ByteArrayInputStream(frame);
IncomingEncryptionLayer i = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypterImpl i = new StreamDecrypterImpl(in, frameCipher,
frameKey, FRAME_LENGTH); frameKey, FRAME_LENGTH);
try { try {
i.readFrame(new byte[FRAME_LENGTH - MAC_LENGTH]); i.readFrame(new byte[MAX_PAYLOAD_LENGTH]);
fail(); fail();
} catch(FormatException expected) {} } catch(FormatException expected) {}
} }
@@ -111,9 +110,9 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
byte[] frame = generateFrame(0, FRAME_LENGTH - 1, 123, true, false); byte[] frame = generateFrame(0, FRAME_LENGTH - 1, 123, true, false);
// Read the frame // Read the frame
ByteArrayInputStream in = new ByteArrayInputStream(frame); ByteArrayInputStream in = new ByteArrayInputStream(frame);
IncomingEncryptionLayer i = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypterImpl i = new StreamDecrypterImpl(in, frameCipher,
frameKey, FRAME_LENGTH); frameKey, FRAME_LENGTH);
int length = i.readFrame(new byte[FRAME_LENGTH - MAC_LENGTH]); int length = i.readFrame(new byte[MAX_PAYLOAD_LENGTH]);
assertEquals(123, length); assertEquals(123, length);
} }
@@ -124,10 +123,10 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
false, false); false, false);
// Try to read the frame, which should fail due to invalid length // Try to read the frame, which should fail due to invalid length
ByteArrayInputStream in = new ByteArrayInputStream(frame); ByteArrayInputStream in = new ByteArrayInputStream(frame);
IncomingEncryptionLayer i = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypterImpl i = new StreamDecrypterImpl(in, frameCipher,
frameKey, FRAME_LENGTH); frameKey, FRAME_LENGTH);
try { try {
i.readFrame(new byte[FRAME_LENGTH - MAC_LENGTH]); i.readFrame(new byte[MAX_PAYLOAD_LENGTH]);
fail(); fail();
} catch(FormatException expected) {} } catch(FormatException expected) {}
} }
@@ -138,10 +137,10 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
byte[] frame = generateFrame(0, FRAME_LENGTH, 123, false, true); byte[] frame = generateFrame(0, FRAME_LENGTH, 123, false, true);
// Try to read the frame, which should fail due to bad padding // Try to read the frame, which should fail due to bad padding
ByteArrayInputStream in = new ByteArrayInputStream(frame); ByteArrayInputStream in = new ByteArrayInputStream(frame);
IncomingEncryptionLayer i = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypterImpl i = new StreamDecrypterImpl(in, frameCipher,
frameKey, FRAME_LENGTH); frameKey, FRAME_LENGTH);
try { try {
i.readFrame(new byte[FRAME_LENGTH - MAC_LENGTH]); i.readFrame(new byte[MAX_PAYLOAD_LENGTH]);
fail(); fail();
} catch(FormatException expected) {} } catch(FormatException expected) {}
} }
@@ -158,12 +157,12 @@ public class IncomingEncryptionLayerTest extends BriarTestCase {
System.arraycopy(frame1, 0, extraFrame, FRAME_LENGTH, FRAME_LENGTH); System.arraycopy(frame1, 0, extraFrame, FRAME_LENGTH, FRAME_LENGTH);
// Read the final frame, which should first read the tag // Read the final frame, which should first read the tag
ByteArrayInputStream in = new ByteArrayInputStream(extraFrame); ByteArrayInputStream in = new ByteArrayInputStream(extraFrame);
IncomingEncryptionLayer i = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypterImpl i = new StreamDecrypterImpl(in, frameCipher,
frameKey, FRAME_LENGTH); frameKey, FRAME_LENGTH);
byte[] buf = new byte[FRAME_LENGTH - MAC_LENGTH]; byte[] payload = new byte[MAX_PAYLOAD_LENGTH];
assertEquals(MAX_PAYLOAD_LENGTH, i.readFrame(buf)); assertEquals(MAX_PAYLOAD_LENGTH, i.readFrame(payload));
// The frame after the final frame should not be read // The frame after the final frame should not be read
assertEquals(-1, i.readFrame(buf)); assertEquals(-1, i.readFrame(payload));
} }
private byte[] generateFrame(long frameNumber, int frameLength, private byte[] generateFrame(long frameNumber, int frameLength,

View File

@@ -1,4 +1,4 @@
package org.briarproject.transport; package org.briarproject.crypto;
import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH; import static org.briarproject.api.transport.TransportConstants.AAD_LENGTH;
import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH; import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;
@@ -15,13 +15,12 @@ import org.briarproject.TestSystemModule;
import org.briarproject.api.crypto.AuthenticatedCipher; import org.briarproject.api.crypto.AuthenticatedCipher;
import org.briarproject.api.crypto.CryptoComponent; import org.briarproject.api.crypto.CryptoComponent;
import org.briarproject.api.crypto.SecretKey; import org.briarproject.api.crypto.SecretKey;
import org.briarproject.crypto.CryptoModule;
import org.junit.Test; import org.junit.Test;
import com.google.inject.Guice; import com.google.inject.Guice;
import com.google.inject.Injector; import com.google.inject.Injector;
public class OutgoingEncryptionLayerTest extends BriarTestCase { public class StreamEncrypterImplTest extends BriarTestCase {
// FIXME: This is an integration test, not a unit test // FIXME: This is an integration test, not a unit test
@@ -30,7 +29,7 @@ public class OutgoingEncryptionLayerTest extends BriarTestCase {
private final CryptoComponent crypto; private final CryptoComponent crypto;
private final AuthenticatedCipher frameCipher; private final AuthenticatedCipher frameCipher;
public OutgoingEncryptionLayerTest() { public StreamEncrypterImplTest() {
Injector i = Guice.createInjector(new CryptoModule(), Injector i = Guice.createInjector(new CryptoModule(),
new TestLifecycleModule(), new TestSystemModule()); new TestLifecycleModule(), new TestSystemModule());
crypto = i.getInstance(CryptoComponent.class); crypto = i.getInstance(CryptoComponent.class);
@@ -52,9 +51,9 @@ public class OutgoingEncryptionLayerTest extends BriarTestCase {
frameCipher.doFinal(plaintext, 0, plaintext.length, ciphertext, 0); frameCipher.doFinal(plaintext, 0, plaintext.length, ciphertext, 0);
// Check that the actual ciphertext matches what's expected // Check that the actual ciphertext matches what's expected
ByteArrayOutputStream out = new ByteArrayOutputStream(); ByteArrayOutputStream out = new ByteArrayOutputStream();
OutgoingEncryptionLayer o = new OutgoingEncryptionLayer(out, StreamEncrypterImpl o = new StreamEncrypterImpl(out,
frameCipher, frameKey, FRAME_LENGTH, null); frameCipher, frameKey, FRAME_LENGTH, null);
o.writeFrame(new byte[FRAME_LENGTH - MAC_LENGTH], payloadLength, false); o.writeFrame(new byte[payloadLength], payloadLength, false);
byte[] actual = out.toByteArray(); byte[] actual = out.toByteArray();
assertEquals(FRAME_LENGTH, actual.length); assertEquals(FRAME_LENGTH, actual.length);
for(int i = 0; i < FRAME_LENGTH; i++) for(int i = 0; i < FRAME_LENGTH; i++)
@@ -78,9 +77,9 @@ public class OutgoingEncryptionLayerTest extends BriarTestCase {
frameCipher.doFinal(plaintext, 0, plaintext.length, ciphertext, 0); frameCipher.doFinal(plaintext, 0, plaintext.length, ciphertext, 0);
// Check that the actual tag and ciphertext match what's expected // Check that the actual tag and ciphertext match what's expected
ByteArrayOutputStream out = new ByteArrayOutputStream(); ByteArrayOutputStream out = new ByteArrayOutputStream();
OutgoingEncryptionLayer o = new OutgoingEncryptionLayer(out, StreamEncrypterImpl o = new StreamEncrypterImpl(out,
frameCipher, frameKey, FRAME_LENGTH, tag); frameCipher, frameKey, FRAME_LENGTH, tag);
o.writeFrame(new byte[FRAME_LENGTH - MAC_LENGTH], payloadLength, false); o.writeFrame(new byte[payloadLength], payloadLength, false);
byte[] actual = out.toByteArray(); byte[] actual = out.toByteArray();
assertEquals(TAG_LENGTH + FRAME_LENGTH, actual.length); assertEquals(TAG_LENGTH + FRAME_LENGTH, actual.length);
for(int i = 0; i < TAG_LENGTH; i++) assertEquals(tag[i], actual[i]); for(int i = 0; i < TAG_LENGTH; i++) assertEquals(tag[i], actual[i]);
@@ -94,7 +93,7 @@ public class OutgoingEncryptionLayerTest extends BriarTestCase {
new Random().nextBytes(tag); new Random().nextBytes(tag);
ByteArrayOutputStream out = new ByteArrayOutputStream(); ByteArrayOutputStream out = new ByteArrayOutputStream();
// Initiator's constructor // Initiator's constructor
OutgoingEncryptionLayer o = new OutgoingEncryptionLayer(out, StreamEncrypterImpl o = new StreamEncrypterImpl(out,
frameCipher, crypto.generateSecretKey(), FRAME_LENGTH, tag); frameCipher, crypto.generateSecretKey(), FRAME_LENGTH, tag);
// Write an empty final frame without having written any other frames // Write an empty final frame without having written any other frames
o.writeFrame(new byte[FRAME_LENGTH - MAC_LENGTH], 0, true); o.writeFrame(new byte[FRAME_LENGTH - MAC_LENGTH], 0, true);

View File

@@ -9,6 +9,8 @@ import static org.briarproject.api.transport.TransportConstants.TAG_LENGTH;
import java.io.ByteArrayInputStream; import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream; import java.io.ByteArrayOutputStream;
import java.io.File; import java.io.File;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.Random; import java.util.Random;
import org.briarproject.BriarTestCase; import org.briarproject.BriarTestCase;
@@ -39,9 +41,7 @@ import org.briarproject.api.messaging.PacketWriter;
import org.briarproject.api.messaging.PacketWriterFactory; import org.briarproject.api.messaging.PacketWriterFactory;
import org.briarproject.api.transport.Endpoint; import org.briarproject.api.transport.Endpoint;
import org.briarproject.api.transport.StreamContext; import org.briarproject.api.transport.StreamContext;
import org.briarproject.api.transport.StreamReader;
import org.briarproject.api.transport.StreamReaderFactory; import org.briarproject.api.transport.StreamReaderFactory;
import org.briarproject.api.transport.StreamWriter;
import org.briarproject.api.transport.StreamWriterFactory; import org.briarproject.api.transport.StreamWriterFactory;
import org.briarproject.api.transport.TagRecogniser; import org.briarproject.api.transport.TagRecogniser;
import org.briarproject.crypto.CryptoModule; import org.briarproject.crypto.CryptoModule;
@@ -136,26 +136,27 @@ public class SimplexMessagingIntegrationTest extends BriarTestCase {
Message message = messageFactory.createAnonymousMessage(null, group, Message message = messageFactory.createAnonymousMessage(null, group,
contentType, timestamp, body); contentType, timestamp, body);
db.addLocalMessage(message); db.addLocalMessage(message);
// Get a stream context
StreamContext ctx = keyManager.getStreamContext(contactId, transportId);
assertNotNull(ctx);
// Create a stream writer // Create a stream writer
ByteArrayOutputStream out = new ByteArrayOutputStream(); ByteArrayOutputStream out = new ByteArrayOutputStream();
StreamWriterFactory streamWriterFactory = StreamWriterFactory streamWriterFactory =
alice.getInstance(StreamWriterFactory.class); alice.getInstance(StreamWriterFactory.class);
StreamContext ctx = keyManager.getStreamContext(contactId, transportId); OutputStream streamWriter = streamWriterFactory.createStreamWriter(out,
assertNotNull(ctx);
StreamWriter streamWriter = streamWriterFactory.createStreamWriter(out,
MAX_FRAME_LENGTH, ctx); MAX_FRAME_LENGTH, ctx);
// Create an outgoing messaging session // Create an outgoing messaging session
EventBus eventBus = alice.getInstance(EventBus.class); EventBus eventBus = alice.getInstance(EventBus.class);
PacketWriterFactory packetWriterFactory = PacketWriterFactory packetWriterFactory =
alice.getInstance(PacketWriterFactory.class); alice.getInstance(PacketWriterFactory.class);
PacketWriter packetWriter = packetWriterFactory.createPacketWriter( PacketWriter packetWriter = packetWriterFactory.createPacketWriter(
streamWriter.getOutputStream()); streamWriter);
MessagingSession session = new SimplexOutgoingSession(db, MessagingSession session = new SimplexOutgoingSession(db,
new ImmediateExecutor(), eventBus, contactId, transportId, new ImmediateExecutor(), eventBus, contactId, transportId,
MAX_LATENCY, packetWriter); MAX_LATENCY, packetWriter);
// Write whatever needs to be written // Write whatever needs to be written
session.run(); session.run();
streamWriter.getOutputStream().close(); streamWriter.close();
// Clean up // Clean up
keyManager.stop(); keyManager.stop();
db.close(); db.close();
@@ -204,7 +205,7 @@ public class SimplexMessagingIntegrationTest extends BriarTestCase {
// Create a stream reader // Create a stream reader
StreamReaderFactory streamReaderFactory = StreamReaderFactory streamReaderFactory =
bob.getInstance(StreamReaderFactory.class); bob.getInstance(StreamReaderFactory.class);
StreamReader streamReader = streamReaderFactory.createStreamReader(in, InputStream streamReader = streamReaderFactory.createStreamReader(in,
MAX_FRAME_LENGTH, ctx); MAX_FRAME_LENGTH, ctx);
// Create an incoming messaging session // Create an incoming messaging session
EventBus eventBus = bob.getInstance(EventBus.class); EventBus eventBus = bob.getInstance(EventBus.class);
@@ -213,7 +214,7 @@ public class SimplexMessagingIntegrationTest extends BriarTestCase {
PacketReaderFactory packetReaderFactory = PacketReaderFactory packetReaderFactory =
bob.getInstance(PacketReaderFactory.class); bob.getInstance(PacketReaderFactory.class);
PacketReader packetReader = packetReaderFactory.createPacketReader( PacketReader packetReader = packetReaderFactory.createPacketReader(
streamReader.getInputStream()); streamReader);
MessagingSession session = new IncomingSession(db, MessagingSession session = new IncomingSession(db,
new ImmediateExecutor(), new ImmediateExecutor(), eventBus, new ImmediateExecutor(), new ImmediateExecutor(), eventBus,
messageVerifier, contactId, transportId, packetReader); messageVerifier, contactId, transportId, packetReader);
@@ -221,7 +222,7 @@ public class SimplexMessagingIntegrationTest extends BriarTestCase {
assertFalse(listener.messageAdded); assertFalse(listener.messageAdded);
// Read whatever needs to be read // Read whatever needs to be read
session.run(); session.run();
streamReader.getInputStream().close(); streamReader.close();
// The private message from Alice should have been added // The private message from Alice should have been added
assertTrue(listener.messageAdded); assertTrue(listener.messageAdded);
// Clean up // Clean up

View File

@@ -4,6 +4,7 @@ import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;
import static org.briarproject.api.transport.TransportConstants.MAC_LENGTH; import static org.briarproject.api.transport.TransportConstants.MAC_LENGTH;
import org.briarproject.BriarTestCase; import org.briarproject.BriarTestCase;
import org.briarproject.api.crypto.StreamDecrypter;
import org.jmock.Expectations; import org.jmock.Expectations;
import org.jmock.Mockery; import org.jmock.Mockery;
import org.junit.Test; import org.junit.Test;
@@ -17,18 +18,18 @@ public class StreamReaderImplTest extends BriarTestCase {
@Test @Test
public void testEmptyFramesAreSkipped() throws Exception { public void testEmptyFramesAreSkipped() throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameReader reader = context.mock(FrameReader.class); final StreamDecrypter decrypter = context.mock(StreamDecrypter.class);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(0)); // Empty frame will(returnValue(0)); // Empty frame
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(2)); // Non-empty frame with two payload bytes will(returnValue(2)); // Non-empty frame with two payload bytes
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(0)); // Empty frame will(returnValue(0)); // Empty frame
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(-1)); // No more frames will(returnValue(-1)); // No more frames
}}); }});
StreamReaderImpl r = new StreamReaderImpl(reader, FRAME_LENGTH); StreamReaderImpl r = new StreamReaderImpl(decrypter, FRAME_LENGTH);
assertEquals(0, r.read()); // Skip the first empty frame, read a byte assertEquals(0, r.read()); // Skip the first empty frame, read a byte
assertEquals(0, r.read()); // Read another byte assertEquals(0, r.read()); // Read another byte
assertEquals(-1, r.read()); // Skip the second empty frame, reach EOF assertEquals(-1, r.read()); // Skip the second empty frame, reach EOF
@@ -40,18 +41,18 @@ public class StreamReaderImplTest extends BriarTestCase {
@Test @Test
public void testEmptyFramesAreSkippedWithBuffer() throws Exception { public void testEmptyFramesAreSkippedWithBuffer() throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameReader reader = context.mock(FrameReader.class); final StreamDecrypter decrypter = context.mock(StreamDecrypter.class);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(0)); // Empty frame will(returnValue(0)); // Empty frame
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(2)); // Non-empty frame with two payload bytes will(returnValue(2)); // Non-empty frame with two payload bytes
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(0)); // Empty frame will(returnValue(0)); // Empty frame
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(-1)); // No more frames will(returnValue(-1)); // No more frames
}}); }});
StreamReaderImpl r = new StreamReaderImpl(reader, FRAME_LENGTH); StreamReaderImpl r = new StreamReaderImpl(decrypter, FRAME_LENGTH);
byte[] buf = new byte[MAX_PAYLOAD_LENGTH]; byte[] buf = new byte[MAX_PAYLOAD_LENGTH];
// Skip the first empty frame, read the two payload bytes // Skip the first empty frame, read the two payload bytes
assertEquals(2, r.read(buf)); assertEquals(2, r.read(buf));
@@ -66,14 +67,14 @@ public class StreamReaderImplTest extends BriarTestCase {
@Test @Test
public void testMultipleReadsPerFrame() throws Exception { public void testMultipleReadsPerFrame() throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameReader reader = context.mock(FrameReader.class); final StreamDecrypter decrypter = context.mock(StreamDecrypter.class);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(MAX_PAYLOAD_LENGTH)); // Nice long frame will(returnValue(MAX_PAYLOAD_LENGTH)); // Nice long frame
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(-1)); // No more frames will(returnValue(-1)); // No more frames
}}); }});
StreamReaderImpl r = new StreamReaderImpl(reader, FRAME_LENGTH); StreamReaderImpl r = new StreamReaderImpl(decrypter, FRAME_LENGTH);
byte[] buf = new byte[MAX_PAYLOAD_LENGTH / 2]; byte[] buf = new byte[MAX_PAYLOAD_LENGTH / 2];
// Read the first half of the payload // Read the first half of the payload
assertEquals(MAX_PAYLOAD_LENGTH / 2, r.read(buf)); assertEquals(MAX_PAYLOAD_LENGTH / 2, r.read(buf));
@@ -88,14 +89,14 @@ public class StreamReaderImplTest extends BriarTestCase {
@Test @Test
public void testMultipleReadsPerFrameWithOffsets() throws Exception { public void testMultipleReadsPerFrameWithOffsets() throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameReader reader = context.mock(FrameReader.class); final StreamDecrypter decrypter = context.mock(StreamDecrypter.class);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(MAX_PAYLOAD_LENGTH)); // Nice long frame will(returnValue(MAX_PAYLOAD_LENGTH)); // Nice long frame
oneOf(reader).readFrame(with(any(byte[].class))); oneOf(decrypter).readFrame(with(any(byte[].class)));
will(returnValue(-1)); // No more frames will(returnValue(-1)); // No more frames
}}); }});
StreamReaderImpl r = new StreamReaderImpl(reader, FRAME_LENGTH); StreamReaderImpl r = new StreamReaderImpl(decrypter, FRAME_LENGTH);
byte[] buf = new byte[MAX_PAYLOAD_LENGTH]; byte[] buf = new byte[MAX_PAYLOAD_LENGTH];
// Read the first half of the payload // Read the first half of the payload
assertEquals(MAX_PAYLOAD_LENGTH / 2, r.read(buf, MAX_PAYLOAD_LENGTH / 2, assertEquals(MAX_PAYLOAD_LENGTH / 2, r.read(buf, MAX_PAYLOAD_LENGTH / 2,

View File

@@ -4,6 +4,7 @@ import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;
import static org.briarproject.api.transport.TransportConstants.MAC_LENGTH; import static org.briarproject.api.transport.TransportConstants.MAC_LENGTH;
import org.briarproject.BriarTestCase; import org.briarproject.BriarTestCase;
import org.briarproject.api.crypto.StreamEncrypter;
import org.jmock.Expectations; import org.jmock.Expectations;
import org.jmock.Mockery; import org.jmock.Mockery;
import org.junit.Test; import org.junit.Test;
@@ -17,15 +18,15 @@ public class StreamWriterImplTest extends BriarTestCase {
@Test @Test
public void testCloseWithoutWritingWritesFinalFrame() throws Exception { public void testCloseWithoutWritingWritesFinalFrame() throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameWriter writer = context.mock(FrameWriter.class); final StreamEncrypter encrypter = context.mock(StreamEncrypter.class);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Write an empty final frame // Write an empty final frame
oneOf(writer).writeFrame(with(any(byte[].class)), with(0), oneOf(encrypter).writeFrame(with(any(byte[].class)), with(0),
with(true)); with(true));
// Flush the stream // Flush the stream
oneOf(writer).flush(); oneOf(encrypter).flush();
}}); }});
StreamWriterImpl w = new StreamWriterImpl(writer, FRAME_LENGTH); StreamWriterImpl w = new StreamWriterImpl(encrypter, FRAME_LENGTH);
w.close(); w.close();
context.assertIsSatisfied(); context.assertIsSatisfied();
} }
@@ -34,14 +35,14 @@ public class StreamWriterImplTest extends BriarTestCase {
public void testFlushWithoutBufferedDataWritesFrameAndFlushes() public void testFlushWithoutBufferedDataWritesFrameAndFlushes()
throws Exception { throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameWriter writer = context.mock(FrameWriter.class); final StreamEncrypter encrypter = context.mock(StreamEncrypter.class);
StreamWriterImpl w = new StreamWriterImpl(writer, FRAME_LENGTH); StreamWriterImpl w = new StreamWriterImpl(encrypter, FRAME_LENGTH);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Write a non-final frame with an empty payload // Write a non-final frame with an empty payload
oneOf(writer).writeFrame(with(any(byte[].class)), with(0), oneOf(encrypter).writeFrame(with(any(byte[].class)), with(0),
with(false)); with(false));
// Flush the stream // Flush the stream
oneOf(writer).flush(); oneOf(encrypter).flush();
}}); }});
w.flush(); w.flush();
context.assertIsSatisfied(); context.assertIsSatisfied();
@@ -49,9 +50,9 @@ public class StreamWriterImplTest extends BriarTestCase {
// Clean up // Clean up
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Closing the writer writes a final frame and flushes again // Closing the writer writes a final frame and flushes again
oneOf(writer).writeFrame(with(any(byte[].class)), with(0), oneOf(encrypter).writeFrame(with(any(byte[].class)), with(0),
with(true)); with(true));
oneOf(writer).flush(); oneOf(encrypter).flush();
}}); }});
w.close(); w.close();
context.assertIsSatisfied(); context.assertIsSatisfied();
@@ -61,14 +62,14 @@ public class StreamWriterImplTest extends BriarTestCase {
public void testFlushWithBufferedDataWritesFrameAndFlushes() public void testFlushWithBufferedDataWritesFrameAndFlushes()
throws Exception { throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameWriter writer = context.mock(FrameWriter.class); final StreamEncrypter encrypter = context.mock(StreamEncrypter.class);
StreamWriterImpl w = new StreamWriterImpl(writer, FRAME_LENGTH); StreamWriterImpl w = new StreamWriterImpl(encrypter, FRAME_LENGTH);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Write a non-final frame with one payload byte // Write a non-final frame with one payload byte
oneOf(writer).writeFrame(with(any(byte[].class)), with(1), oneOf(encrypter).writeFrame(with(any(byte[].class)), with(1),
with(false)); with(false));
// Flush the stream // Flush the stream
oneOf(writer).flush(); oneOf(encrypter).flush();
}}); }});
w.write(0); w.write(0);
w.flush(); w.flush();
@@ -77,9 +78,9 @@ public class StreamWriterImplTest extends BriarTestCase {
// Clean up // Clean up
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Closing the writer writes a final frame and flushes again // Closing the writer writes a final frame and flushes again
oneOf(writer).writeFrame(with(any(byte[].class)), with(0), oneOf(encrypter).writeFrame(with(any(byte[].class)), with(0),
with(true)); with(true));
oneOf(writer).flush(); oneOf(encrypter).flush();
}}); }});
w.close(); w.close();
context.assertIsSatisfied(); context.assertIsSatisfied();
@@ -88,11 +89,11 @@ public class StreamWriterImplTest extends BriarTestCase {
@Test @Test
public void testSingleByteWritesWriteFullFrame() throws Exception { public void testSingleByteWritesWriteFullFrame() throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameWriter writer = context.mock(FrameWriter.class); final StreamEncrypter encrypter = context.mock(StreamEncrypter.class);
StreamWriterImpl w = new StreamWriterImpl(writer, FRAME_LENGTH); StreamWriterImpl w = new StreamWriterImpl(encrypter, FRAME_LENGTH);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Write a full non-final frame // Write a full non-final frame
oneOf(writer).writeFrame(with(any(byte[].class)), oneOf(encrypter).writeFrame(with(any(byte[].class)),
with(MAX_PAYLOAD_LENGTH), with(false)); with(MAX_PAYLOAD_LENGTH), with(false));
}}); }});
for(int i = 0; i < MAX_PAYLOAD_LENGTH; i++) { for(int i = 0; i < MAX_PAYLOAD_LENGTH; i++) {
@@ -103,9 +104,9 @@ public class StreamWriterImplTest extends BriarTestCase {
// Clean up // Clean up
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Closing the writer writes a final frame and flushes again // Closing the writer writes a final frame and flushes again
oneOf(writer).writeFrame(with(any(byte[].class)), with(0), oneOf(encrypter).writeFrame(with(any(byte[].class)), with(0),
with(true)); with(true));
oneOf(writer).flush(); oneOf(encrypter).flush();
}}); }});
w.close(); w.close();
context.assertIsSatisfied(); context.assertIsSatisfied();
@@ -114,11 +115,11 @@ public class StreamWriterImplTest extends BriarTestCase {
@Test @Test
public void testMultiByteWritesWriteFullFrames() throws Exception { public void testMultiByteWritesWriteFullFrames() throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameWriter writer = context.mock(FrameWriter.class); final StreamEncrypter encrypter = context.mock(StreamEncrypter.class);
StreamWriterImpl w = new StreamWriterImpl(writer, FRAME_LENGTH); StreamWriterImpl w = new StreamWriterImpl(encrypter, FRAME_LENGTH);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Write two full non-final frames // Write two full non-final frames
exactly(2).of(writer).writeFrame(with(any(byte[].class)), exactly(2).of(encrypter).writeFrame(with(any(byte[].class)),
with(MAX_PAYLOAD_LENGTH), with(false)); with(MAX_PAYLOAD_LENGTH), with(false));
}}); }});
// Sanity check // Sanity check
@@ -134,9 +135,9 @@ public class StreamWriterImplTest extends BriarTestCase {
// Clean up // Clean up
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Closing the writer writes a final frame and flushes again // Closing the writer writes a final frame and flushes again
oneOf(writer).writeFrame(with(any(byte[].class)), with(0), oneOf(encrypter).writeFrame(with(any(byte[].class)), with(0),
with(true)); with(true));
oneOf(writer).flush(); oneOf(encrypter).flush();
}}); }});
w.close(); w.close();
context.assertIsSatisfied(); context.assertIsSatisfied();
@@ -145,17 +146,17 @@ public class StreamWriterImplTest extends BriarTestCase {
@Test @Test
public void testLargeMultiByteWriteWritesFullFrames() throws Exception { public void testLargeMultiByteWriteWritesFullFrames() throws Exception {
Mockery context = new Mockery(); Mockery context = new Mockery();
final FrameWriter writer = context.mock(FrameWriter.class); final StreamEncrypter encrypter = context.mock(StreamEncrypter.class);
StreamWriterImpl w = new StreamWriterImpl(writer, FRAME_LENGTH); StreamWriterImpl w = new StreamWriterImpl(encrypter, FRAME_LENGTH);
context.checking(new Expectations() {{ context.checking(new Expectations() {{
// Write two full non-final frames // Write two full non-final frames
exactly(2).of(writer).writeFrame(with(any(byte[].class)), exactly(2).of(encrypter).writeFrame(with(any(byte[].class)),
with(MAX_PAYLOAD_LENGTH), with(false)); with(MAX_PAYLOAD_LENGTH), with(false));
// Write a final frame with a one-byte payload // Write a final frame with a one-byte payload
oneOf(writer).writeFrame(with(any(byte[].class)), with(1), oneOf(encrypter).writeFrame(with(any(byte[].class)), with(1),
with(true)); with(true));
// Flush the stream // Flush the stream
oneOf(writer).flush(); oneOf(encrypter).flush();
}}); }});
// Write two full payloads using one large multi-byte write // Write two full payloads using one large multi-byte write
byte[] b = new byte[MAX_PAYLOAD_LENGTH * 2 + 1]; byte[] b = new byte[MAX_PAYLOAD_LENGTH * 2 + 1];

View File

@@ -0,0 +1,44 @@
package org.briarproject.transport;
import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;
import static org.briarproject.api.transport.TransportConstants.MAC_LENGTH;
import java.io.EOFException;
import java.io.IOException;
import java.io.InputStream;
import org.briarproject.api.FormatException;
import org.briarproject.api.crypto.StreamDecrypter;
import org.briarproject.util.ByteUtils;
class TestStreamDecrypter implements StreamDecrypter {
private final InputStream in;
private final byte[] frame;
TestStreamDecrypter(InputStream in, int frameLength) {
this.in = in;
frame = new byte[frameLength];
}
public int readFrame(byte[] payload) throws IOException {
int offset = 0;
while(offset < HEADER_LENGTH) {
int read = in.read(frame, offset, HEADER_LENGTH - offset);
if(read == -1) throw new EOFException();
offset += read;
}
boolean finalFrame = (frame[0] & 0x80) == 0x80;
int payloadLength = ByteUtils.readUint16(frame, 0) & 0x7FFF;
while(offset < frame.length) {
int read = in.read(frame, offset, frame.length - offset);
if(read == -1) break;
offset += read;
}
if(!finalFrame && offset < frame.length) throw new EOFException();
if(offset < HEADER_LENGTH + payloadLength + MAC_LENGTH)
throw new FormatException();
System.arraycopy(frame, HEADER_LENGTH, payload, 0, payloadLength);
return payloadLength;
}
}

View File

@@ -0,0 +1,44 @@
package org.briarproject.transport;
import static org.briarproject.api.transport.TransportConstants.HEADER_LENGTH;
import static org.briarproject.api.transport.TransportConstants.MAC_LENGTH;
import java.io.IOException;
import java.io.OutputStream;
import org.briarproject.api.crypto.StreamEncrypter;
import org.briarproject.util.ByteUtils;
class TestStreamEncrypter implements StreamEncrypter {
private final OutputStream out;
private final byte[] tag, frame;
private boolean writeTag = true;
TestStreamEncrypter(OutputStream out, int frameLength, byte[] tag) {
this.out = out;
this.tag = tag;
frame = new byte[frameLength];
}
public void writeFrame(byte[] payload, int payloadLength,
boolean finalFrame) throws IOException {
if(writeTag) {
out.write(tag);
writeTag = false;
}
ByteUtils.writeUint16(payloadLength, frame, 0);
if(finalFrame) frame[0] |= 0x80;
System.arraycopy(payload, 0, frame, HEADER_LENGTH, payloadLength);
for(int i = HEADER_LENGTH + payloadLength; i < frame.length; i++)
frame[i] = 0;
if(finalFrame)
out.write(frame, 0, HEADER_LENGTH + payloadLength + MAC_LENGTH);
else out.write(frame, 0, frame.length);
}
public void flush() throws IOException {
out.flush();
}
}

View File

@@ -11,48 +11,18 @@ import java.io.OutputStream;
import java.util.Random; import java.util.Random;
import org.briarproject.BriarTestCase; import org.briarproject.BriarTestCase;
import org.briarproject.TestLifecycleModule; import org.briarproject.api.crypto.StreamDecrypter;
import org.briarproject.TestSystemModule; import org.briarproject.api.crypto.StreamEncrypter;
import org.briarproject.api.crypto.AuthenticatedCipher;
import org.briarproject.api.crypto.CryptoComponent;
import org.briarproject.api.crypto.SecretKey;
import org.briarproject.api.transport.StreamWriterFactory;
import org.briarproject.crypto.CryptoModule;
import org.junit.Test; import org.junit.Test;
import com.google.inject.AbstractModule;
import com.google.inject.Guice;
import com.google.inject.Injector;
import com.google.inject.Module;
public class TransportIntegrationTest extends BriarTestCase { public class TransportIntegrationTest extends BriarTestCase {
private final int FRAME_LENGTH = 2048; private final int FRAME_LENGTH = 2048;
private final CryptoComponent crypto;
private final AuthenticatedCipher frameCipher;
private final Random random; private final Random random;
private final byte[] secret;
private final SecretKey tagKey, frameKey;
public TransportIntegrationTest() { public TransportIntegrationTest() {
Module testModule = new AbstractModule() {
@Override
public void configure() {
bind(StreamWriterFactory.class).to(
StreamWriterFactoryImpl.class);
}
};
Injector i = Guice.createInjector(testModule, new CryptoModule(),
new TestLifecycleModule(), new TestSystemModule());
crypto = i.getInstance(CryptoComponent.class);
frameCipher = crypto.getFrameCipher();
random = new Random(); random = new Random();
// Since we're sending frames to ourselves, we only need outgoing keys
secret = new byte[32];
random.nextBytes(secret);
tagKey = crypto.deriveTagKey(secret, true);
frameKey = crypto.deriveFrameKey(secret, 0, true);
} }
@Test @Test
@@ -66,27 +36,24 @@ public class TransportIntegrationTest extends BriarTestCase {
} }
private void testWriteAndRead(boolean initiator) throws Exception { private void testWriteAndRead(boolean initiator) throws Exception {
// Encode the tag // Generate a random tag
byte[] tag = new byte[TAG_LENGTH]; byte[] tag = new byte[TAG_LENGTH];
crypto.encodeTag(tag, tagKey, 0); random.nextBytes(tag);
// Generate two random frames // Generate two frames with random payloads
byte[] frame = new byte[1234]; byte[] payload1 = new byte[1234];
random.nextBytes(frame); random.nextBytes(payload1);
byte[] frame1 = new byte[321]; byte[] payload2 = new byte[321];
random.nextBytes(frame1); random.nextBytes(payload2);
// Copy the frame key - the copy will be erased
SecretKey frameCopy = frameKey.copy();
// Write the tag and the frames // Write the tag and the frames
ByteArrayOutputStream out = new ByteArrayOutputStream(); ByteArrayOutputStream out = new ByteArrayOutputStream();
FrameWriter frameWriter = new OutgoingEncryptionLayer(out, StreamEncrypter encrypter = new TestStreamEncrypter(out, FRAME_LENGTH,
frameCipher, frameCopy, FRAME_LENGTH, tag); tag);
StreamWriterImpl streamWriter = new StreamWriterImpl(frameWriter, OutputStream streamWriter = new StreamWriterImpl(encrypter,
FRAME_LENGTH); FRAME_LENGTH);
OutputStream out1 = streamWriter.getOutputStream(); streamWriter.write(payload1);
out1.write(frame); streamWriter.flush();
out1.flush(); streamWriter.write(payload2);
out1.write(frame1); streamWriter.flush();
out1.flush();
byte[] output = out.toByteArray(); byte[] output = out.toByteArray();
assertEquals(TAG_LENGTH + FRAME_LENGTH * 2, output.length); assertEquals(TAG_LENGTH + FRAME_LENGTH * 2, output.length);
// Read the tag back // Read the tag back
@@ -95,17 +62,15 @@ public class TransportIntegrationTest extends BriarTestCase {
read(in, recoveredTag); read(in, recoveredTag);
assertArrayEquals(tag, recoveredTag); assertArrayEquals(tag, recoveredTag);
// Read the frames back // Read the frames back
FrameReader frameReader = new IncomingEncryptionLayer(in, frameCipher, StreamDecrypter decrypter = new TestStreamDecrypter(in, FRAME_LENGTH);
frameKey, FRAME_LENGTH); InputStream streamReader = new StreamReaderImpl(decrypter,
StreamReaderImpl streamReader = new StreamReaderImpl(frameReader,
FRAME_LENGTH); FRAME_LENGTH);
InputStream in1 = streamReader.getInputStream(); byte[] recoveredPayload1 = new byte[payload1.length];
byte[] recoveredFrame = new byte[frame.length]; read(streamReader, recoveredPayload1);
read(in1, recoveredFrame); assertArrayEquals(payload1, recoveredPayload1);
assertArrayEquals(frame, recoveredFrame); byte[] recoveredPayload2 = new byte[payload2.length];
byte[] recoveredFrame1 = new byte[frame1.length]; read(streamReader, recoveredPayload2);
read(in1, recoveredFrame1); assertArrayEquals(payload2, recoveredPayload2);
assertArrayEquals(frame1, recoveredFrame1);
streamWriter.close(); streamWriter.close();
streamReader.close(); streamReader.close();
} }