⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 multiringnode.java

📁 pastry的java实现的2.0b版
💻 JAVA
字号:
/*************************************************************************"FreePastry" Peer-to-Peer Application Development Substrate Copyright 2002, Rice University. All rights reserved.Redistribution and use in source and binary forms, with or withoutmodification, are permitted provided that the following conditions aremet:- Redistributions of source code must retain the above copyrightnotice, this list of conditions and the following disclaimer.- Redistributions in binary form must reproduce the above copyrightnotice, this list of conditions and the following disclaimer in thedocumentation and/or other materials provided with the distribution.- Neither  the name  of Rice  University (RICE) nor  the names  of itscontributors may be  used to endorse or promote  products derived fromthis software without specific prior written permission.This software is provided by RICE and the contributors on an "as is"basis, without any representations or warranties of any kind, expressor implied including, but not limited to, representations orwarranties of non-infringement, merchantability or fitness for aparticular purpose. In no event shall RICE or contributors be liablefor any direct, indirect, incidental, special, exemplary, orconsequential damages (including, but not limited to, procurement ofsubstitute goods or services; loss of use, data, or profits; orbusiness interruption) however caused and on any theory of liability,whether in contract, strict liability, or tort (including negligenceor otherwise) arising in any way out of the use of this software, evenif advised of the possibility of such damage.********************************************************************************/package rice.p2p.multiring;import java.io.IOException;import java.util.*;import rice.environment.Environment;import rice.environment.logging.Logger;import rice.p2p.commonapi.*;import rice.p2p.commonapi.rawserialization.*;import rice.p2p.multiring.messaging.*;import rice.p2p.scribe.*;import rice.p2p.scribe.rawserialization.ScribeContentDeserializer;/** * @(#) MultiringNode.java This class wraps a Node, enabling it to support * multiple hierarchitcal scoped rings. In order to use the mulitring * functionality, only one change is necessary: when one constructs a Node, one * should change Node test = factory.newNode(...); to Node test = new * MultiringNode(factory.newNode(...)); * * @version $Id: MultiringNode.java 3274 2006-05-15 16:17:47Z jeffh $ * @author Alan Mislove */public class MultiringNode implements Node, ScribeClient {  /**   * The node which this mulitring node is wrapping   */  protected Node node;  /**   * The Id which represents the current ring this node is a member of   */  protected Id ringId;  /**   * The Scribe application which the node uses to do routing   */  protected Scribe scribe;  /**   * The collection, which keeps track of the other nodes on the ring node   */  protected MultiringNodeCollection collection;  /**   * The list of all of the endpoints connected to this node   */  protected Hashtable endpoints;  /**   * A cached IdFactory for internal use   */  protected MultiringIdFactory factory;  /**   * The environment   */  protected Environment environment;  /**   * DESCRIBE THE FIELD   */  protected Logger logger;  /**   * Constructor   *   * @param node The node which this multiring node is wrapping   * @param ringId The Id of this node's ring   */  public MultiringNode(Id ringId, Node node) {    this.node = node;    this.environment = node.getEnvironment();    this.logger = environment.getLogManager().getLogger(MultiringNode.class, null);    this.ringId = ringId;    this.endpoints = new Hashtable();    this.scribe = new ScribeImpl(this, "Multiring");    scribe.setContentDeserializer(      new ScribeContentDeserializer() {        public ScribeContent deserializeScribeContent(InputBuffer buf, Endpoint endpoint, short type) throws IOException {          switch (type) {            case RingMessage.TYPE:              return new RingMessage(buf, endpoint, endpoints);          }          throw new IllegalArgumentException("Invalid type:" + type);        }      });    this.collection = new MultiringNodeCollection(this, environment.getParameters().getInt("p2p_multiring_base"));    this.factory = (MultiringIdFactory) getIdFactory();  }  /**   * Constructor   *   * @param node The node which this multiring node is wrapping   * @param ringId The Id of this node's ring   * @param existing An existing node which this node should pair with   */  public MultiringNode(Id ringId, Node node, MultiringNode existing) {    this(ringId, node);    this.collection = existing.getCollection();    this.collection.addNode(this);  }  /**   * Gets the LocalNodeHandle attribute of the MultiringNode object   *   * @return The LocalNodeHandle value   */  public NodeHandle getLocalNodeHandle() {    return new MultiringNodeHandle(getRingId(), node.getLocalNodeHandle());  }  /**   * This returns a Endpoint specific to the given application and instance name   * to the application, which the application can then use in order to send an   * receive messages. This method allows advanced developers to specify which   * "port" on the node they wish their application to register as. This "port"   * determines which of the applications on top of the node should receive an   * incoming message. NOTE: Use of this method of registering applications is   * recommended only for advanced users - 99% of all applications should just   * use the other registerApplication   *   * @return The endpoint specific to this applicationk, which can be used for   *      message sending/receiving.   *///  public Endpoint registerApplication(Application application, int port) {//    Endpoint endpoint = new MultiringEndpoint(this, node.registerApplication(new MultiringApplication(getRingId(), application), port), application);//    endpoints.put(endpoint.getInstance(), endpoint);////    return endpoint;//  }  /**   * Returns the Id of this node   *   * @return This node's Id   */  public Id getId() {    return RingId.build(ringId, node.getId());  }  /**   * Returns the ringId of this node   *   * @return This node's ringId   */  public Id getRingId() {    return ringId;  }  /**   * Returns the underlying id of this node   *   * @return This node's Id   */  public Id getNodeId() {    return node.getId();  }  /**   * Returns this mutliring node's internal node   *   * @return The wrapped node   */  public Node getNode() {    return node;  }  /**   * Returns the collection this node is a member of   *   * @return This node's collection   */  public MultiringNodeCollection getCollection() {    return collection;  }  /**   * Returns a factory for Ids specific to this node's protocol.   *   * @return A factory for creating Ids.   */  public IdFactory getIdFactory() {    return new MultiringIdFactory(ringId, node.getIdFactory());  }  /**   * Implements the ring routing mechanism at the local node. Thus, if the   * target ringId has our ringId as a prefix, we route to substring(id,   * length+1). Otherwise, we route to substring(getId(), length-1).   *   * @param id The target   * @return The ringId we should route to   */  private Id getTarget(RingId id) {    int shared = collection.getLengthOfSharedPrefix((RingId) getId(), id);    int thisLength = collection.getLength((RingId) getId());    int targetLength = collection.getLength(id);    if (shared == thisLength) {      return makeTarget(id, thisLength + 1);    } else {      return makeTarget((RingId) getId(), thisLength - 1);    }  }  /**   * Getter for the environment.   *   * @return the environment   */  public Environment getEnvironment() {    return environment;  }  /**   * This returns a VirtualizedNode specific to the given application and   * instance name to the application, which the application can then use in   * order to send an receive messages.   *   * @param application The Application   * @param instance An identifier for a given instance   * @return The endpoint specific to this applicationk, which can be used for   *      message sending/receiving.   */  public Endpoint registerApplication(Application application, String instance) {    Endpoint endpoint = new MultiringEndpoint(this, node.buildEndpoint(new MultiringApplication(getRingId(), application), application.getClass() + "-" + instance), application);    endpoints.put(endpoint.getInstance(), endpoint);    endpoint.register();    return endpoint;  }  /**   * DESCRIBE THE METHOD   *   * @param application DESCRIBE THE PARAMETER   * @param instance DESCRIBE THE PARAMETER   * @return DESCRIBE THE RETURN VALUE   */  public Endpoint buildEndpoint(Application application, String instance) {    Endpoint endpoint = new MultiringEndpoint(this, node.buildEndpoint(new MultiringApplication(getRingId(), application), application.getClass() + "-" + instance), application);    endpoints.put(endpoint.getInstance(), endpoint);    return endpoint;  }  /**   * Method which is used to inform the node that another node has been added to   * the collection. This node then joins the appropriate Scribe group to   * advertise this it can route directly to that node's ring.   *   * @param otherRingId DESCRIBE THE PARAMETER   */  protected void nodeAdded(Id otherRingId) {    //System.outt.println("JOINING SCRIBE GROUP " + otherRingId + " AT NODE " + getId());    scribe.subscribe(new Topic(RingId.build(ringId, otherRingId)), this);  }  /**   * This method implements the ring routing algortihm attempting to find a   * suitable nextHop for the message. First, this method checks to see if there   * is a local node which can get the message closer to the target. If there is   * no such node, this method anycasts the message to the scribe group for the   * next ring in the heirarchy.   *   * @param id The destination Id of the message.   * @param message The message to deliver   * @param application DESCRIBE THE PARAMETER   */  void route(RingId id, RawMessage message, String application) {    if (id.getRingId().equals(ringId)) {      MultiringEndpoint endpoint = (MultiringEndpoint) endpoints.get(application);      endpoint.route(id, message, null);    } else {      //System.outt.println("ANYCASTING TO SCRIBE GROUP " + getTarget(id) + " AT NODE " + getId() + " FOR APPLICATION " + application);      scribe.anycast(new Topic(RingId.build(ringId, getTarget(id))), new RingMessage(id, message, application));    }  }  /**   * Utility method for building a target id of specified length - all remaining   * bytes are zeroed out   *   * @param id The target id   * @param length The length in bytes   * @return The target   */  private Id makeTarget(RingId id, int length) {    byte[] current = id.getRingId().toByteArray();    byte[] bytes = new byte[current.length];    for (int j = 0; j < collection.BASE * length; j++) {      bytes[bytes.length - 1 - j] = current[bytes.length - 1 - j];    }    return factory.buildNormalId(bytes);  }  /**   * This method is invoked when an anycast is received for a topic which this   * client is interested in. The client should return whether or not the   * anycast should continue.   *   * @param topic The topic the message was anycasted to   * @param content The content which was anycasted   * @return Whether or not the anycast should continue   */  public boolean anycast(Topic topic, ScribeContent content) {    if (content instanceof RingMessage) {      RingMessage rm = (RingMessage) content;      //System.outt.println("RECEIVED ANYCAST TO " + rm.getId() + " AT NODE " + getId());      collection.route(rm.getId(), rm.getMessage(), rm.getApplication());    } else {      if (logger.level <= Logger.WARNING) {        logger.log(          "Received unrecognized message " + content);      }    }    return true;  }  /**   * This method is invoked when a message is delivered for a topic this client   * is interested in.   *   * @param topic The topic the message was published to   * @param content The content which was published   */  public void deliver(Topic topic, ScribeContent content) {    if (logger.level <= Logger.FINER) {      logger.log(        "Received unexpected delivery on topic " + topic + " of " + content);    }  }  /**   * Informs this client that a child was added to a topic in which it was   * interested in.   *   * @param topic The topic to unsubscribe from   * @param child The child that was added   */  public void childAdded(Topic topic, NodeHandle child) {  }  /**   * Informs this client that a child was removed from a topic in which it was   * interested in.   *   * @param topic The topic to unsubscribe from   * @param child The child that was removed   */  public void childRemoved(Topic topic, NodeHandle child) {  }  /**   * Informs the client that a subscribe on the given topic failed - the client   * should retry the subscribe or take appropriate action.   *   * @param topic The topic which the subscribe failed on   */  public void subscribeFailed(Topic topic) {    if (logger.level <= Logger.WARNING) {      logger.log(        getId() + ": Received error joining ringId topic " + topic + " - trying again.");    }    nodeAdded(((RingId) topic.getId()).getId());  }  /**   * Prints out the string   *   * @return A string   */  public String toString() {    return "{MultiringNode " + getId() + "}";  }}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -