Package com.aelitis.azureus.core.peermanager.messaging.azureus

Source Code of com.aelitis.azureus.core.peermanager.messaging.azureus.AZHandshake

/*
* 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 == nullthrow 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 == nullthrow new MessageException("mid == null");
      ids[pos] = new String(mid);

      byte[] ver = (byte[]) msg.get("ver");
      if (ver == nullthrow new MessageException("ver == null");
     
      if (ver.length != 1throw 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();
  }
}
TOP

Related Classes of com.aelitis.azureus.core.peermanager.messaging.azureus.AZHandshake

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.