| 
 | Apache Tomcat 6.0.45 | ||||||||
| PREV CLASS NEXT CLASS | FRAMES NO FRAMES | ||||||||
| SUMMARY: NESTED | FIELD | CONSTR | METHOD | DETAIL: FIELD | CONSTR | METHOD | ||||||||
java.lang.Objectorg.apache.catalina.tribes.group.ChannelInterceptorBase
org.apache.catalina.tribes.group.interceptors.NonBlockingCoordinator
public class NonBlockingCoordinator
Title: Auto merging leader election algorithm
Description: Implementation of a simple coordinator algorithm that not only selects a coordinator, it also merges groups automatically when members are discovered that werent part of the
This algorithm is non blocking meaning it allows for transactions while the coordination phase is going on
This implementation is based on a home brewed algorithm that uses the AbsoluteOrder of a membership
 to pass a token ring of the current membership.
 This is not the same as just using AbsoluteOrder! Consider the following scenario:
 Nodes, A,B,C,D,E on a network, in that priority. AbsoluteOrder will only work if all
 nodes are receiving pings from all the other nodes. 
 meaning, that node{i} receives pings from node{all}-node{i}
 but the following could happen if a multicast problem occurs.
 A has members {B,C,D}
 B has members {A,C}
 C has members {D,E}
 D has members {A,B,C,E}
 E has members {A,C,D}
 Because the default Tribes membership implementation, relies on the multicast packets to 
 arrive at all nodes correctly, there is nothing guaranteeing that it will.
 
 To best explain how this algorithm works, lets take the above example:
 For simplicity we assume that a send operation is O(1) for all nodes, although this algorithm will work
 where messages overlap, as they all depend on absolute order
 Scenario 1: A,B,C,D,E all come online at the same time
 Eval phase, A thinks of itself as leader, B thinks of A as leader,
 C thinks of itself as leader, D,E think of A as leader
 Token phase:
 (1) A sends out a message X{A-ldr, A-src, mbrs-A,B,C,D} to B where X is the id for the message(and the view)
 (1) C sends out a message Y{C-ldr, C-src, mbrs-C,D,E} to D where Y is the id for the message(and the view)
 (2) B receives X{A-ldr, A-src, mbrs-A,B,C,D}, sends X{A-ldr, A-src, mbrs-A,B,C,D} to C 
 (2) D receives Y{C-ldr, C-src, mbrs-C,D,E} D is aware of A,B, sends Y{A-ldr, C-src, mbrs-A,B,C,D,E} to E
 (3) C receives X{A-ldr, A-src, mbrs-A,B,C,D}, sends X{A-ldr, A-src, mbrs-A,B,C,D,E} to D
 (3) E receives Y{A-ldr, C-src, mbrs-A,B,C,D,E} sends Y{A-ldr, C-src, mbrs-A,B,C,D,E} to A
 (4) D receives X{A-ldr, A-src, mbrs-A,B,C,D,E} sends sends X{A-ldr, A-src, mbrs-A,B,C,D,E} to A
 (4) A receives Y{A-ldr, C-src, mbrs-A,B,C,D,E}, holds the message, add E to its list of members
 (5) A receives X{A-ldr, A-src, mbrs-A,B,C,D,E} 
 At this point, the state looks like
 A - {A-ldr, mbrs-A,B,C,D,E, id=X}
 B - {A-ldr, mbrs-A,B,C,D, id=X}
 C - {A-ldr, mbrs-A,B,C,D,E, id=X}
 D - {A-ldr, mbrs-A,B,C,D,E, id=X}
 E - {A-ldr, mbrs-A,B,C,D,E, id=Y}
 
 A message doesn't stop until it reaches its original sender, unless its dropped by a higher leader.
 As you can see, E still thinks the viewId=Y, which is not correct. But at this point we have 
 arrived at the same membership and all nodes are informed of each other.
 To synchronize the rest we simply perform the following check at A when A receives X:
 Original X{A-ldr, A-src, mbrs-A,B,C,D} == Arrived X{A-ldr, A-src, mbrs-A,B,C,D,E}
 Since the condition is false, A, will resend the token, and A sends X{A-ldr, A-src, mbrs-A,B,C,D,E} to B
 When A receives X again, the token is complete. 
 Optionally, A can send a message X{A-ldr, A-src, mbrs-A,B,C,D,E confirmed} to A,B,C,D,E who then
 install and accept the view.
 
 Lets assume that C1 arrives, C1 has lower priority than C, but higher priority than D.
 Lets also assume that C1 sees the following view {B,D,E}
 C1 waits for a token to arrive. When the token arrives, the same scenario as above will happen.
 In the scenario where C1 sees {D,E} and A,B,C can not see C1, no token will ever arrive.
 In this case, C1 sends a Z{C1-ldr, C1-src, mbrs-C1,D,E} to D
 D receives Z{C1-ldr, C1-src, mbrs-C1,D,E} and sends Z{A-ldr, C1-src, mbrs-A,B,C,C1,D,E} to E
 E receives Z{A-ldr, C1-src, mbrs-A,B,C,C1,D,E} and sends it to A
 A sends Z{A-ldr, A-src, mbrs-A,B,C,C1,D,E} to B and the chain continues until A receives the token again.
 At that time A optionally sends out Z{A-ldr, A-src, mbrs-A,B,C,C1,D,E, confirmed} to A,B,C,C1,D,E
 
To ensure that the view gets implemented at all nodes at the same time, A will send out a VIEW_CONF message, this is the 'confirmed' message that is optional above.
Ideally, the interceptor below this one would be the TcpFailureDetector to ensure correct memberships
The example above, of course can be simplified with a finite statemachine:
 But I suck at writing state machines, my head gets all confused. One day I will document this algorithm though.
 Maybe I'll do a state diagram :)
 
| Nested Class Summary | |
|---|---|
| static class | NonBlockingCoordinator.CoordinationEvent | 
| static class | NonBlockingCoordinator.CoordinationMessage | 
| Nested classes/interfaces inherited from interface org.apache.catalina.tribes.ChannelInterceptor | 
|---|
| ChannelInterceptor.InterceptorEvent | 
| Field Summary | |
|---|---|
| protected static byte[] | COORD_ALIVEAlive message | 
| protected static byte[] | COORD_CONFCoordination confirmation, for blocking installations | 
| protected static byte[] | COORD_HEADERheader for a coordination message | 
| protected static byte[] | COORD_REQUESTCoordination request | 
| protected  java.util.concurrent.atomic.AtomicBoolean | coordMsgReceived | 
| protected  java.lang.Object | electionMutex | 
| protected  Membership | membershipOur nonblocking membership | 
| protected  boolean | started | 
| protected  int | startsvc | 
| protected  Membership | suggestedView | 
| protected  UniqueId | suggestedviewIdindicates that we are running an election and this is the one we are running | 
| protected  Membership | viewOur current view | 
| protected  UniqueId | viewIdOut current viewId | 
| protected  long | waitForCoordMsgTimeoutTime to wait for coordination timeout | 
| Fields inherited from class org.apache.catalina.tribes.group.ChannelInterceptorBase | 
|---|
| log, optionFlag | 
| Constructor Summary | |
|---|---|
| NonBlockingCoordinator() | |
| Method Summary | |
|---|---|
|  boolean | accept(ChannelMessage msg) | 
| protected  boolean | alive(Member mbr) | 
|  ChannelData | createData(NonBlockingCoordinator.CoordinationMessage msg,
           MemberImpl local) | 
|  void | fireInterceptorEvent(ChannelInterceptor.InterceptorEvent event) | 
|  Member | getCoordinator()Returns coordinator if one is available | 
|  Member | getLocalMember(boolean incAlive)Return the member that represents this node. | 
|  Member | getMember(Member mbr)Intercepts the code>Channel.getMember(Member) method | 
|  Member[] | getMembers()Get all current cluster members | 
|  Member | getNextInLine(MemberImpl local,
              MemberImpl[] others) | 
|  Member[] | getView() | 
|  UniqueId | getViewId() | 
| protected  void | halt()Block in/out messages while a election is going on | 
| protected  void | handleMyToken(MemberImpl local,
              NonBlockingCoordinator.CoordinationMessage msg,
              Member sender,
              Membership merged) | 
| protected  void | handleOtherToken(MemberImpl local,
                 NonBlockingCoordinator.CoordinationMessage msg,
                 Member sender,
                 Membership merged) | 
| protected  void | handleToken(NonBlockingCoordinator.CoordinationMessage msg,
            Member sender,
            Membership merged) | 
| protected  void | handleViewConf(NonBlockingCoordinator.CoordinationMessage msg,
               Member sender,
               Membership merged) | 
| protected  boolean | hasHigherPriority(Member[] complete,
                  Member[] local) | 
|  boolean | hasMembers()has members | 
|  void | heartbeat()The heartbeat()method gets invoked periodically
 to allow interceptors to clean up resources, time out object and 
 perform actions that are unrelated to sending/receiving data. | 
|  boolean | isCoordinator() | 
|  boolean | isHighest() | 
| protected  boolean | isViewConf(NonBlockingCoordinator.CoordinationMessage msg) | 
|  void | memberAdded(Member member)A member was added to the group | 
|  void | memberAdded(Member member,
            boolean elect) | 
|  void | memberDisappeared(Member member)A member was removed from the group If the member left voluntarily, the Member.getCommand will contain the Member.SHUTDOWN_PAYLOAD data | 
| protected  Membership | mergeOnArrive(NonBlockingCoordinator.CoordinationMessage msg,
              Member sender) | 
|  void | messageReceived(ChannelMessage msg)the messageReceivedis invoked when a message is received. | 
| protected  void | processCoordMessage(NonBlockingCoordinator.CoordinationMessage msg,
                    Member sender) | 
| protected  void | release()Release lock for in/out messages election is completed | 
| protected  void | sendElectionMsg(MemberImpl local,
                MemberImpl next,
                NonBlockingCoordinator.CoordinationMessage msg) | 
| protected  void | sendElectionMsgToNextInline(MemberImpl local,
                            NonBlockingCoordinator.CoordinationMessage msg) | 
|  void | sendMessage(Member[] destination,
            ChannelMessage msg,
            InterceptorPayload payload)The sendMessagemethod is called when a message is being sent to one more destinations. | 
| protected  void | setupMembership() | 
|  void | start(int svc)Starts up the channel. | 
|  void | startElection(boolean force) | 
|  void | stop(int svc)Shuts down the channel. | 
| protected  void | viewChange(UniqueId viewId,
           Member[] view) | 
| protected  void | waitForRelease()Wait for an election to end | 
| Methods inherited from class org.apache.catalina.tribes.group.ChannelInterceptorBase | 
|---|
| getNext, getOptionFlag, getPrevious, okToProcess, setNext, setOptionFlag, setPrevious | 
| Methods inherited from class java.lang.Object | 
|---|
| clone, equals, finalize, getClass, hashCode, notify, notifyAll, toString, wait, wait, wait | 
| Field Detail | 
|---|
protected static final byte[] COORD_HEADER
protected static final byte[] COORD_REQUEST
protected static final byte[] COORD_CONF
protected static final byte[] COORD_ALIVE
protected long waitForCoordMsgTimeout
protected Membership view
protected UniqueId viewId
protected Membership membership
protected UniqueId suggestedviewId
protected Membership suggestedView
protected boolean started
protected final int startsvc
protected java.lang.Object electionMutex
protected java.util.concurrent.atomic.AtomicBoolean coordMsgReceived
| Constructor Detail | 
|---|
public NonBlockingCoordinator()
| Method Detail | 
|---|
public void startElection(boolean force)
                   throws ChannelException
ChannelException
protected void sendElectionMsg(MemberImpl local,
                               MemberImpl next,
                               NonBlockingCoordinator.CoordinationMessage msg)
                        throws ChannelException
ChannelException
protected void sendElectionMsgToNextInline(MemberImpl local,
                                           NonBlockingCoordinator.CoordinationMessage msg)
                                    throws ChannelException
ChannelException
public Member getNextInLine(MemberImpl local,
                            MemberImpl[] others)
public ChannelData createData(NonBlockingCoordinator.CoordinationMessage msg,
                              MemberImpl local)
protected void viewChange(UniqueId viewId,
                          Member[] view)
protected boolean alive(Member mbr)
protected Membership mergeOnArrive(NonBlockingCoordinator.CoordinationMessage msg,
                                   Member sender)
protected void processCoordMessage(NonBlockingCoordinator.CoordinationMessage msg,
                                   Member sender)
                            throws ChannelException
ChannelException
protected void handleToken(NonBlockingCoordinator.CoordinationMessage msg,
                           Member sender,
                           Membership merged)
                    throws ChannelException
ChannelException
protected void handleMyToken(MemberImpl local,
                             NonBlockingCoordinator.CoordinationMessage msg,
                             Member sender,
                             Membership merged)
                      throws ChannelException
ChannelException
protected void handleOtherToken(MemberImpl local,
                                NonBlockingCoordinator.CoordinationMessage msg,
                                Member sender,
                                Membership merged)
                         throws ChannelException
ChannelException
protected void handleViewConf(NonBlockingCoordinator.CoordinationMessage msg,
                              Member sender,
                              Membership merged)
                       throws ChannelException
ChannelExceptionprotected boolean isViewConf(NonBlockingCoordinator.CoordinationMessage msg)
protected boolean hasHigherPriority(Member[] complete,
                                    Member[] local)
public Member getCoordinator()
public Member[] getView()
public UniqueId getViewId()
protected void halt()
protected void release()
protected void waitForRelease()
public void start(int svc)
           throws ChannelException
ChannelInterceptorBase
start in interface ChannelInterceptorstart in class ChannelInterceptorBasesvc - int value of ChannelException - if a startup error occurs or the service is already started.Channel
public void stop(int svc)
          throws ChannelException
ChannelInterceptorBase
stop in interface ChannelInterceptorstop in class ChannelInterceptorBasesvc - int value of ChannelException - if a startup error occurs or the service is already started.Channel
public void sendMessage(Member[] destination,
                        ChannelMessage msg,
                        InterceptorPayload payload)
                 throws ChannelException
ChannelInterceptorsendMessage method is called when a message is being sent to one more destinations.
 The interceptor can modify any of the parameters and then pass on the message down the stack by
 invoking getNext().sendMessage(destination,msg,payload)getNext().sendMessage(destination,msg,payload)
sendMessage in interface ChannelInterceptorsendMessage in class ChannelInterceptorBasedestination - Member[] - the destination for this messagemsg - ChannelMessage - the message to be sentpayload - InterceptorPayload - the payload, carrying an error handler and future useful data, can be null
ChannelExceptionErrorHandler, 
InterceptorPayloadpublic void messageReceived(ChannelMessage msg)
ChannelInterceptormessageReceived is invoked when a message is received.
 ChannelMessage.getAddress() is the sender, or the reply-to address
 if it has been overwritten.
messageReceived in interface ChannelInterceptormessageReceived in class ChannelInterceptorBasemsg - ChannelMessagepublic boolean accept(ChannelMessage msg)
accept in class ChannelInterceptorBasepublic void memberAdded(Member member)
MembershipListener
memberAdded in interface MembershipListenermemberAdded in class ChannelInterceptorBasemember - Member - the member that was added
public void memberAdded(Member member,
                        boolean elect)
public void memberDisappeared(Member member)
MembershipListener
memberDisappeared in interface MembershipListenermemberDisappeared in class ChannelInterceptorBasemember - MemberMember.SHUTDOWN_PAYLOADpublic boolean isHighest()
public boolean isCoordinator()
public void heartbeat()
ChannelInterceptorheartbeat() method gets invoked periodically
 to allow interceptors to clean up resources, time out object and 
 perform actions that are unrelated to sending/receiving data.
heartbeat in interface ChannelInterceptorheartbeat in interface Heartbeatheartbeat in class ChannelInterceptorBasepublic boolean hasMembers()
hasMembers in interface ChannelInterceptorhasMembers in class ChannelInterceptorBaseChannel.hasMembers()public Member[] getMembers()
getMembers in interface ChannelInterceptorgetMembers in class ChannelInterceptorBaseChannel.getMembers()public Member getMember(Member mbr)
ChannelInterceptor
getMember in interface ChannelInterceptorgetMember in class ChannelInterceptorBasembr - Member
Channel.getMember(Member)public Member getLocalMember(boolean incAlive)
getLocalMember in interface ChannelInterceptorgetLocalMember in class ChannelInterceptorBaseincAlive - boolean
Channel.getLocalMember(boolean)protected void setupMembership()
public void fireInterceptorEvent(ChannelInterceptor.InterceptorEvent event)
fireInterceptorEvent in interface ChannelInterceptorfireInterceptorEvent in class ChannelInterceptorBase| 
 | Apache Tomcat 6.0.45 | ||||||||
| PREV CLASS NEXT CLASS | FRAMES NO FRAMES | ||||||||
| SUMMARY: NESTED | FIELD | CONSTR | METHOD | DETAIL: FIELD | CONSTR | METHOD | ||||||||