All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.axis2.clustering.tribes.MembershipManager Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License. You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied. See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */

package org.apache.axis2.clustering.tribes;

import org.apache.axis2.clustering.ClusteringConstants;
import org.apache.axis2.clustering.control.wka.MemberJoinedCommand;
import org.apache.axis2.clustering.control.wka.MemberListCommand;
import org.apache.axis2.clustering.management.GroupManagementAgent;
import org.apache.axis2.context.ConfigurationContext;
import org.apache.catalina.tribes.Channel;
import org.apache.catalina.tribes.Member;
import org.apache.catalina.tribes.RemoteProcessException;
import org.apache.catalina.tribes.group.Response;
import org.apache.catalina.tribes.group.RpcChannel;
import org.apache.catalina.tribes.group.interceptors.StaticMembershipInterceptor;
import org.apache.catalina.tribes.membership.MemberImpl;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

/**
 * Responsible for managing the membership. Handles membership changes.
 */
public class MembershipManager {

    private static final Log log = LogFactory.getLog(MembershipManager.class);

    private RpcChannel rpcMembershipChannel;
    private StaticMembershipInterceptor staticMembershipInterceptor;

    /**
     * The domain corresponding to the membership handled by this MembershipManager
     */
    private byte[] domain;
    private GroupManagementAgent groupManagementAgent;
    private ConfigurationContext configContext;


    /**
     * List of current members in the cluster. Only the members who are alive will be in this
     * list
     */
    private final List members = new ArrayList();

    /**
     * List of Well-Known members. These members may or may not be alive at a given moment.
     */
    private final List wkaMembers = new ArrayList();

    /**
     * List of Well-Known members which have not responded to the MEMBER_LIST message.
     * We need to retry sending the MEMBER_LIST message to these members until they respond,
     * otherwise, we cannot be sure whether these WKA members added the members in the MEMBER_LIST
     */
    private final List nonRespondingWkaMembers = new CopyOnWriteArrayList();

    /**
     * The member representing this node
     */
    private Member localMember;

    /**
     *
     */
    private boolean isMemberListResponseReceived;

    public MembershipManager(ConfigurationContext configContext) {
        this.configContext = configContext;
    }

    public MembershipManager() {
    }

    public void setRpcMembershipChannel(RpcChannel rpcMembershipChannel) {
        this.rpcMembershipChannel = rpcMembershipChannel;
    }

    public RpcChannel getRpcMembershipChannel() {
        return rpcMembershipChannel;
    }

    public void setupStaticMembershipManagement(StaticMembershipInterceptor staticMembershipInterceptor) {
        this.staticMembershipInterceptor = staticMembershipInterceptor;
        ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
        scheduler.scheduleWithFixedDelay(new MemberListSenderTask(), 5, 5, TimeUnit.SECONDS);
    }

    public void setGroupManagementAgent(GroupManagementAgent groupManagementAgent) {
        this.groupManagementAgent = groupManagementAgent;
    }

    public void setDomain(byte[] domain) {
        this.domain = domain;
    }

    public byte[] getDomain() {
        return domain;
    }

    public Member getLocalMember() {
        return localMember;
    }

    public void setLocalMember(Member localMember) {
        this.localMember = localMember;
    }

    public void addWellKnownMember(Member wkaMember) {
        wkaMembers.add(wkaMember);
    }

    public void removeWellKnownMember(Member wkaMember) {
        wkaMembers.remove(wkaMember);
    }

    /**
     * A new member is added
     *
     * @param member The new member that joined the cluster
     * @return true  If the member was added to the members array; false, otherwise.
     */
    public boolean memberAdded(Member member) {

        if (log.isDebugEnabled()) {
            log.debug("members.contains(member) =" + members.contains(member));
            log.debug("Is in my domain: " + TribesUtil.isInDomain(member, domain));
        }

        // If this member already exists or if the member belongs to another domain,
        // there is no need to add it
        if (members.contains(member) || !TribesUtil.isInDomain(member, domain)) {
            return false;
        }

        if (staticMembershipInterceptor != null) { // this interceptor is null when multicast based scheme is used
            staticMembershipInterceptor.addStaticMember(member);
            if (log.isDebugEnabled()) {
                log.debug("Added static member " + TribesUtil.getName(member));
            }
        }

        boolean shouldAddMember = localMember == null ||
                                  TribesUtil.areInSameDomain(localMember, member);

        // If this member is a load balancer, notify the respective load balance event handler?
        if (groupManagementAgent != null) {
            log.info("Application member " + TribesUtil.getName(member) + " joined group " +
                     new String(member.getDomain()));
            groupManagementAgent.applicationMemberAdded(TribesUtil.toAxis2Member(member));
        }

        if (shouldAddMember) {
            boolean wkaMemberBelongsToLocalDomain = true;
            if (rpcMembershipChannel != null && isLocalMemberInitialized() &&
                wkaMembers.contains(member)) { // if it is a well-known member

                log.info("A WKA member " + TribesUtil.getName(member) +
                         " just joined the group. Sending MEMBER_LIST message.");
                wkaMemberBelongsToLocalDomain = sendMemberListToWellKnownMember(member);
            }
            if (wkaMemberBelongsToLocalDomain) {
                members.add(member);
                if (log.isDebugEnabled()) {
                    log.debug("Added group member " + TribesUtil.getName(member) + " to domain " +
                              new String(member.getDomain()));
                }
                return true;
            }
        }
        return false;
    }

    /**
     * Task which send MEMBER_LIST messages to WKA members which have not yet responded to the
     * MEMBER_LIST message
     */
    private class MemberListSenderTask implements Runnable {
        public void run() {
            try {
                if (nonRespondingWkaMembers != null && !nonRespondingWkaMembers.isEmpty()) {
                    for (Member wkaMember : nonRespondingWkaMembers) {
                        if (wkaMember != null) {
                            sendMemberListToWellKnownMember(wkaMember);
                        }
                    }
                }
            } catch (Throwable e) {
                log.error("Could not send MemberList to WKA Members", e);
            }
        }
    }

    /**
     * Send MEMBER_LIST message to WKA member
     *
     * @param wkaMember The WKA member to whom the MEMBER_LIST has to be sent
     * @return true - if the WKA member belongs to the domain of this local member
     */
    private boolean sendMemberListToWellKnownMember(Member wkaMember) {
        /*if (wkaMember.isFailing() || wkaMember.isSuspect()) {
            return false;
        }*/
        // send the member list to it
        MemberListCommand memListCmd;
        try {
            memListCmd = new MemberListCommand();
            List members = new ArrayList(this.members);
            members.add(localMember); // Need to set the local member too
            memListCmd.setMembers(members.toArray(new Member[members.size()]));

            Response[] responses =
                    rpcMembershipChannel.send(new Member[]{wkaMember}, memListCmd,
                                              RpcChannel.ALL_REPLY,
                                              Channel.SEND_OPTIONS_ASYNCHRONOUS |
                                              TribesConstants.MEMBERSHIP_MSG_OPTION, 10000);

            // Once a response is received from the WKA member to the MEMBER_LIST message,
            // if it does not belong to this domain, simply remove it from the members
            if (responses != null && responses.length > 0 && responses[0] != null) {
                nonRespondingWkaMembers.remove(wkaMember);
                Member source = responses[0].getSource();
                if (!TribesUtil.areInSameDomain(source, wkaMember)) {
                    if (log.isDebugEnabled()) {
                        log.debug("WKA Member " + TribesUtil.getName(source) +
                                  " does not belong to local domain " + new String(domain) +
                                  ". Hence removing it from the list.");
                    }
                    return false;
                }
            } else { // No response from WKA member
                nonRespondingWkaMembers.add(wkaMember);
            }
        } catch (Exception e) {
            String errMsg = "Could not send MEMBER_LIST to well-known member " +
                            TribesUtil.getName(wkaMember);
            log.error(errMsg, e);
            throw new RemoteProcessException(errMsg, e);
        }
        return true;
    }

    /**
     * Send the list of members to the member
     *
     * @param member The member to whom the member list has to be sent
     */
    public void sendMemberList(Member member) {
        try {
            MemberListCommand memListCmd = new MemberListCommand();
            List members = new ArrayList(this.members);
            memListCmd.setMembers(members.toArray(new Member[members.size()]));
            rpcMembershipChannel.send(new Member[]{member}, memListCmd, RpcChannel.ALL_REPLY,
                                      Channel.SEND_OPTIONS_ASYNCHRONOUS |
                                      TribesConstants.MEMBERSHIP_MSG_OPTION, 10000);
            if (log.isDebugEnabled()) {
                log.debug("Sent MEMBER_LIST to " + TribesUtil.getName(member));
            }
        } catch (Exception e) {
            String errMsg = "Could not send MEMBER_LIST to member " + TribesUtil.getName(member);
            log.error(errMsg, e);
            throw new RemoteProcessException(errMsg, e);
        }
    }

    /**
     * Inform all members that a particular member just joined
     *
     * @param member The member who just joined
     */
    public void sendMemberJoinedToAll(Member member) {
        try {

            MemberJoinedCommand cmd = new MemberJoinedCommand();
            cmd.setMember(member);
            ArrayList membersToSend = (ArrayList) (((ArrayList) members).clone());
            membersToSend.remove(member); // Do not send MEMBER_JOINED to the new member who just joined

            if (membersToSend.size() > 0) {
                rpcMembershipChannel.send(membersToSend.toArray(new Member[membersToSend.size()]), cmd,
                                          RpcChannel.ALL_REPLY,
                                          Channel.SEND_OPTIONS_ASYNCHRONOUS |
                                          TribesConstants.MEMBERSHIP_MSG_OPTION,
                                          10000);
                if (log.isDebugEnabled()) {
                    log.debug("Sent MEMBER_JOINED[" + TribesUtil.getName(member) +
                              "] to all members in domain " + new String(domain));
                }
            }
        } catch (Exception e) {
            String errMsg = "Could not send MEMBER_JOINED[" + TribesUtil.getName(member) +
                            "] to all members ";
            log.error(errMsg, e);
            throw new RemoteProcessException(errMsg, e);
        }
    }

    private boolean isLocalMemberInitialized() {
        if (configContext == null) {
            return false;
        }
        Object clusterInitialized =
                configContext.getPropertyNonReplicable(ClusteringConstants.CLUSTER_INITIALIZED);
        return clusterInitialized != null && clusterInitialized.equals("true");
    }

    /**
     * A member disappeared
     *
     * @param member The member that left the cluster
     */
    public void memberDisappeared(Member member) {
        members.remove(member);
        nonRespondingWkaMembers.remove(member);

        // Is this an application domain member?
        if (groupManagementAgent != null) {
            groupManagementAgent.applicationMemberRemoved(TribesUtil.toAxis2Member(member));
        }
    }

    /**
     * Get the list of current members
     *
     * @return list of current members
     */
    public Member[] getMembers() {
        return members.toArray(new Member[members.size()]);
    }

    /**
     * Get the member that has been alive for the longest time
     *
     * @return The member that has been alive for the longest time
     */
    public Member getLongestLivingMember() {
        Member longestLivingMember = null;
        if (members.size() > 0) {
            Member member0 = members.get(0);
            long longestAliveTime = member0.getMemberAliveTime();
            longestLivingMember = member0;
            for (Member member : members) {
                if (longestAliveTime < member.getMemberAliveTime()) {
                    longestAliveTime = member.getMemberAliveTime();
                    longestLivingMember = member;
                }
            }
        }
        return longestLivingMember;
    }

    /**
     * Get a random member from the list of current members
     *
     * @return A random member from the list of current members
     */
    public Member getRandomMember() {
        if (members.size() == 0) {
            return null;
        }
        int memberIndex = new Random().nextInt(members.size());
        return members.get(memberIndex);
    }

    /**
     * Check whether there are any members
     *
     * @return true if there are other members, false otherwise
     */
    public boolean hasMembers() {
        return members.size() > 0;
    }

    /**
     * Get a member
     *
     * @param member The member to be found
     * @return The member, if it is found
     */
    public Member getMember(Member member) {
        if (hasMembers()) {
            MemberImpl result = null;
            for (int i = 0; i < this.members.size() && result == null; i++) {
                if (members.get(i).equals(member)) {
                    result = (MemberImpl) members.get(i);
                }
            }
            return result;
        }
        return null;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy