/*
* Created on Apr 30, 2004
* Created by Alon Rohter
* Copyright (C) 2004, 2005, 2006 Aelitis, All Rights Reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* AELITIS, SAS au capital de 46,603.30 euros
* 8 Allee Lenotre, La Grille Royale, 78600 Le Mesnil le Roi, France.
*
*/
package com.aelitis.azureus.core.peermanager.messaging.azureus;
import java.net.InetAddress;
import java.util.*;
import org.gudy.azureus2.core3.util.ByteFormatter;
import org.gudy.azureus2.core3.util.Debug;
import org.gudy.azureus2.core3.util.DirectByteBuffer;
import org.gudy.azureus2.core3.util.HashWrapper;
import com.aelitis.azureus.core.peermanager.messaging.Message;
import com.aelitis.azureus.core.peermanager.messaging.MessageException;
import com.aelitis.azureus.core.peermanager.messaging.MessagingUtil;
/**
* AZ handshake message.
*/
public class AZHandshake implements AZMessage {
public static final int HANDSHAKE_TYPE_PLAIN = 0;
public static final int HANDSHAKE_TYPE_CRYPTO = 1;
private static final byte bss = DirectByteBuffer.SS_MSG;
private final byte version;
private DirectByteBuffer buffer = null;
private String description = null;
private final byte[] identity;
private final HashWrapper sessionID;
private final HashWrapper reconnectID;
private final String client;
private final String client_version;
private final String[] avail_ids;
private final byte[] avail_versions;
private int tcp_port;
private int udp_port;
private int udp_non_data_port;
private final int handshake_type;
private final boolean uploadOnly;
private final InetAddress ipv6;
public AZHandshake( byte[] peer_identity,
HashWrapper sessionID,
HashWrapper reconnectID,
String _client,
String version,
int tcp_listen_port,
int udp_listen_port,
int udp_non_data_listen_port,
InetAddress ipv6addr,
String[] avail_msg_ids,
byte[] avail_msg_versions,
int _handshake_type,
byte _version,
boolean uploadOnly) {
this.identity = peer_identity;
this.sessionID = sessionID;
this.reconnectID = reconnectID;
this.client = _client;
this.client_version = version;
this.avail_ids = avail_msg_ids;
this.avail_versions = avail_msg_versions;
this.tcp_port = tcp_listen_port;
this.udp_port = udp_listen_port;
this.udp_non_data_port = udp_non_data_listen_port;
this.handshake_type = _handshake_type;
this.version = _version;
this.uploadOnly = uploadOnly;
this.ipv6 = ipv6addr;
//verify given port info is ok
if( tcp_port < 0 || tcp_port > 65535 ) {
Debug.out( "given TCP listen port is invalid: " +tcp_port );
tcp_port = 0;
}
if( udp_port < 0 || udp_port > 65535 ) {
Debug.out( "given UDP listen port is invalid: " +udp_port );
udp_port = 0;
}
if( udp_non_data_port < 0 || udp_non_data_port > 65535 ) {
Debug.out( "given UDP non-data listen port is invalid: " +udp_non_data_port );
udp_non_data_port = 0;
}
}
public byte[] getIdentity() { return identity; }
public HashWrapper getRemoteSessionID() { return sessionID; }
public HashWrapper getReconnectSessionID() { return reconnectID; }
public boolean isUploadOnly() {return uploadOnly;}
public String getClient() { return client; }
public String getClientVersion() { return client_version; }
public String[] getMessageIDs() { return avail_ids; }
public byte[] getMessageVersions() { return avail_versions; }
public int getTCPListenPort() { return tcp_port; }
public int getUDPListenPort() { return udp_port; }
public int getUDPNonDataListenPort() { return udp_non_data_port; }
public InetAddress getIPv6() { return ipv6; }
public int getHandshakeType() { return handshake_type; }
public String getID() { return AZMessage.ID_AZ_HANDSHAKE; }
public byte[] getIDBytes() { return AZMessage.ID_AZ_HANDSHAKE_BYTES; }
public String getFeatureID() { return AZMessage.AZ_FEATURE_ID; }
public int getFeatureSubID() { return AZMessage.SUBID_AZ_HANDSHAKE; }
public int getType() { return Message.TYPE_PROTOCOL_PAYLOAD; }
public byte getVersion() { return version; };
public String getDescription() {
if( description == null ) {
String msgs_desc = "";
for( int i=0; i < avail_ids.length; i++ ) {
String id = avail_ids[ i ];
byte ver = avail_versions[ i ];
if( id.equals( getID() ) ) continue; //skip ourself
msgs_desc += "[" +id+ ":" +ver+ "]";
}
description = getID()+ " from [" +ByteFormatter.nicePrint( identity, true )+ ", " +
client+ " " +client_version+ ", TCP/UDP ports " +tcp_port+ "/" +udp_port+ "/" + udp_non_data_port +
", handshake " + (getHandshakeType() == HANDSHAKE_TYPE_PLAIN ? "plain" : "crypto") +
", upload_only = " + (isUploadOnly() ? "1" : "0") +
(ipv6 != null ? ", ipv6 = "+ipv6.getHostAddress() : "") +
(sessionID != null ? ", sessionID: "+sessionID.toBase32String() : "") +
(reconnectID != null ? ", reconnect request: "+reconnectID.toBase32String() : "") +
"] supports " +msgs_desc;
}
return description;
}
public DirectByteBuffer[] getData() {
if (buffer == null)
{
Map payload_map = new HashMap();
//client info
payload_map.put("identity", identity);
if (sessionID != null)
payload_map.put("session", sessionID.getBytes());
if (reconnectID != null)
payload_map.put("reconn", reconnectID.getBytes());
payload_map.put("client", client);
payload_map.put("version", client_version);
payload_map.put("tcp_port", new Long(tcp_port));
payload_map.put("udp_port", new Long(udp_port));
payload_map.put("udp2_port", new Long(udp_non_data_port));
payload_map.put("handshake_type", new Long(handshake_type));
payload_map.put("upload_only", new Long(uploadOnly ? 1L : 0L));
if(ipv6 != null)
payload_map.put("ipv6", ipv6.getAddress());
//available message list
List message_list = new ArrayList();
for (int i = 0; i < avail_ids.length; i++)
{
String id = avail_ids[i];
byte ver = avail_versions[i];
if (id.equals(getID()))
continue; //skip ourself
Map msg = new HashMap();
msg.put("id", id);
msg.put("ver", new byte[] { ver });
message_list.add(msg);
}
payload_map.put("messages", message_list);
// random padding if crypto
if (handshake_type == AZHandshake.HANDSHAKE_TYPE_CRYPTO)
payload_map.put("pad", new byte[(int) (Math.random() * AZMessageFactory.AZ_HANDSHAKE_PAD_MAX)]);
buffer = MessagingUtil.convertPayloadToBencodedByteStream(payload_map, DirectByteBuffer.AL_MSG_AZ_HAND);
if (buffer.remaining(bss) > 1200)
System.out.println("Generated AZHandshake size = " + buffer.remaining(bss) + " bytes");
}
return new DirectByteBuffer[] { buffer };
}
public Message deserialize( DirectByteBuffer data, byte version ) throws MessageException {
Map root = MessagingUtil.convertBencodedByteStreamToPayload( data, 100, getID() );
byte[] id = (byte[])root.get( "identity" );
if( id == null ) throw new MessageException( "id == null" );
if( id.length != 20 ) throw new MessageException( "id.length != 20: " +id.length );
byte[] session = (byte[])root.get("session");
byte[] reconnect = (byte[])root.get("reconn");
byte[] raw_name = (byte[])root.get( "client" );
if( raw_name == null ) throw new MessageException( "raw_name == null" );
String name = new String( raw_name );
byte[] raw_ver = (byte[])root.get( "version" );
if( raw_ver == null ) throw new MessageException( "raw_ver == null" );
String client_version = new String( raw_ver );
Long tcp_lport = (Long)root.get( "tcp_port" );
if( tcp_lport == null ) { //old handshake
tcp_lport = new Long( 0 );
}
Long udp_lport = (Long)root.get( "udp_port" );
if( udp_lport == null ) { //old handshake
udp_lport = new Long( 0 );
}
Long udp2_lport = (Long)root.get( "udp2_port" );
if( udp2_lport == null ) { //old handshake
udp2_lport = udp_lport;
}
Long h_type = (Long)root.get( "handshake_type" );
if( h_type == null ) { //only 2307+ send type
h_type = new Long( HANDSHAKE_TYPE_PLAIN );
}
InetAddress ipv6 = null;
if(root.get("ipv6") instanceof byte[])
{
try
{
InetAddress.getByAddress((byte[]) root.get("ipv6"));
} catch (Exception e)
{
}
}
List raw_msgs = (List) root.get("messages");
if (raw_msgs == null) throw new MessageException("raw_msgs == null");
String[] ids = new String[raw_msgs.size()];
byte[] vers = new byte[raw_msgs.size()];
int pos = 0;
for (Iterator i = raw_msgs.iterator(); i.hasNext();) {
Map msg = (Map) i.next();
byte[] mid = (byte[]) msg.get("id");
if (mid == null) throw new MessageException("mid == null");
ids[pos] = new String(mid);
byte[] ver = (byte[]) msg.get("ver");
if (ver == null) throw new MessageException("ver == null");
if (ver.length != 1) throw new MessageException("ver.length != 1");
vers[pos] = ver[0];
pos++;
}
Long ulOnly = (Long)root.get("upload_only");
boolean uploadOnly = ulOnly != null && ulOnly.longValue() > 0L ? true : false;
return new AZHandshake( id, session == null ? null : new HashWrapper(session),reconnect == null ? null : new HashWrapper(reconnect), name, client_version, tcp_lport.intValue(), udp_lport.intValue(), udp2_lport.intValue(), ipv6 ,ids, vers, h_type.intValue(), version , uploadOnly);
}
public void destroy() {
if( buffer != null ) buffer.returnToPool();
}
}