All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.i2p.data.i2np.UnknownI2NPMessage Maven / Gradle / Ivy

The newest version!
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("size mismatch, too big, 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(getUniqueId());
        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();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy