/* * 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.util.*; import org.gudy.azureus2.core3.util.*; 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; 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, String[] avail_msg_ids, byte[] avail_msg_versions, int _handshake_type, byte _version ) { 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; //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 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 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") + (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 ) ); //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 ); } 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++; } 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(), ids, vers, h_type.intValue(), version ); } public void destroy() { if( buffer != null ) buffer.returnToPool(); } }