forked from I2P_Developers/i2p.i2p
- Deprecate unused stream methods and I2NPMessageReader since all transports provide encapsulation. - Don't throw IOE from byte array methods - Use cached null cert in GarlicClove - Add method to limit size of buffer to read - Don't check checksum at input, in most cases - Reuse checksum at output, for unomodified pass-through messages (but recalculating it now and logging on a mismatch for testing) - Fix DatabaseLookupMessage to internally store the don't include peers as a List, not a Set, so it doesn't get reordered and break the checksum - Log cleanup * NTCP: - Zero-copy and limit size when handing buffer to I2NP - Log hex dump message on I2NPMessageException, like in SSU - Don't close connection on I2NPMessageException
133 lines
4.8 KiB
Java
133 lines
4.8 KiB
Java
package net.i2p.data.i2np;
|
|
/*
|
|
* free (adj.): unencumbered; not under the control of others
|
|
* Written by jrandom in 2003 and released into the public domain
|
|
* with no warranty of any kind, either expressed or implied.
|
|
* It probably won't make your computer catch on fire, or eat
|
|
* your children, but it might. Use at your own risk.
|
|
*
|
|
*/
|
|
|
|
import net.i2p.I2PAppContext;
|
|
import net.i2p.data.DataHelper;
|
|
import net.i2p.data.Hash;
|
|
import net.i2p.util.SimpleByteCache;
|
|
|
|
/**
|
|
* This is similar to DataMessage or GarlicMessage but with a variable message type.
|
|
* This is defined so routers can route messages they don't know about.
|
|
* We don't extend those classes so that any code that does (instanceof foo)
|
|
* won't return true for this type. Load tests use DataMessage, for example.
|
|
* Also, those classes include an additional length field that we can't use here.
|
|
* See InboundMessageDistributor.
|
|
*
|
|
* There is no setData() method, the only way to create one of these is to
|
|
* read it with readMessage() (i.e., it came from some other router)
|
|
*
|
|
* As of 0.8.12 this class is working. It is used at the IBGW to reduce the processing
|
|
* required. For zero-hop IB tunnels, the convert() method is used to reconstitute
|
|
* a standard message class.
|
|
*
|
|
* @since 0.7.12 but broken before 0.8.12
|
|
*/
|
|
public class UnknownI2NPMessage extends FastI2NPMessageImpl {
|
|
private byte _data[];
|
|
private final int _type;
|
|
|
|
/** @param type 0-255 */
|
|
public UnknownI2NPMessage(I2PAppContext context, int type) {
|
|
super(context);
|
|
_type = type;
|
|
}
|
|
|
|
/**
|
|
* @throws IllegalStateException if data previously set, to protect saved checksum
|
|
*/
|
|
public void readMessage(byte data[], int offset, int dataSize, int type) throws I2NPMessageException {
|
|
if (_data != null)
|
|
throw new IllegalStateException();
|
|
if (type != _type) throw new I2NPMessageException("Message type is incorrect for this message");
|
|
if (dataSize > MAX_SIZE)
|
|
throw new I2NPMessageException("wtf, size=" + dataSize);
|
|
_data = new byte[dataSize];
|
|
System.arraycopy(data, offset, _data, 0, dataSize);
|
|
}
|
|
|
|
/** calculate the message body's length (not including the header and footer */
|
|
protected int calculateWrittenLength() {
|
|
if (_data == null)
|
|
return 0;
|
|
else
|
|
return _data.length;
|
|
}
|
|
|
|
/** write the message body to the output array, starting at the given index */
|
|
protected int writeMessageBody(byte out[], int curIndex) {
|
|
if (_data != null) {
|
|
System.arraycopy(_data, 0, out, curIndex, _data.length);
|
|
curIndex += _data.length;
|
|
}
|
|
return curIndex;
|
|
}
|
|
|
|
/**
|
|
* Note that this returns the "true" type, so that
|
|
* the IBGW can correctly make drop decisions.
|
|
*
|
|
* @return 0-255
|
|
*/
|
|
public int getType() { return _type; }
|
|
|
|
/**
|
|
* Attempt to convert this message to a known message class.
|
|
* This does the delayed verification using the saved checksum.
|
|
*
|
|
* Used by TunnelGatewayZeroHop.
|
|
*
|
|
* @throws I2NPMessageException if the conversion fails
|
|
* @since 0.8.12
|
|
*/
|
|
public I2NPMessage convert() throws I2NPMessageException {
|
|
if (_data == null || !_hasChecksum)
|
|
throw new I2NPMessageException("Illegal state");
|
|
I2NPMessage msg = I2NPMessageImpl.createMessage(_context, _type);
|
|
if (msg instanceof UnknownI2NPMessage)
|
|
throw new I2NPMessageException("Unable to convert unknown type " + _type);
|
|
byte[] calc = SimpleByteCache.acquire(Hash.HASH_LENGTH);
|
|
_context.sha().calculateHash(_data, 0, _data.length, calc, 0);
|
|
boolean eq = _checksum == calc[0];
|
|
SimpleByteCache.release(calc);
|
|
if (!eq)
|
|
throw new I2NPMessageException("Bad checksum on " + _data.length + " byte msg type " + _type);
|
|
msg.readMessage(_data, 0, _data.length, _type);
|
|
msg.setUniqueId(_uniqueId);
|
|
msg.setMessageExpiration(_expiration);
|
|
return msg;
|
|
}
|
|
|
|
@Override
|
|
public int hashCode() {
|
|
return _type + DataHelper.hashCode(_data);
|
|
}
|
|
|
|
@Override
|
|
public boolean equals(Object object) {
|
|
if ( (object != null) && (object instanceof UnknownI2NPMessage) ) {
|
|
UnknownI2NPMessage msg = (UnknownI2NPMessage)object;
|
|
return _type == msg.getType() && DataHelper.eq(_data, msg._data);
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
StringBuilder buf = new StringBuilder();
|
|
buf.append("[UnknownI2NPMessage: ");
|
|
buf.append("\n\tType: ").append(_type);
|
|
buf.append("\n\tLength: ").append(calculateWrittenLength());
|
|
buf.append("]");
|
|
return buf.toString();
|
|
}
|
|
}
|