Package com.alibaba.otter.shared.arbitrate.impl.setl.monitor

Source Code of com.alibaba.otter.shared.arbitrate.impl.setl.monitor.NodeMonitor

/*
* Copyright (C) 2010-2101 Alibaba Group Holding Limited.
*
* Licensed 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 com.alibaba.otter.shared.arbitrate.impl.setl.monitor;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.ExecutorService;

import org.I0Itec.zkclient.IZkChildListener;
import org.I0Itec.zkclient.exception.ZkNodeExistsException;
import org.apache.commons.lang.ClassUtils;
import org.apache.commons.lang.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.alibaba.otter.shared.arbitrate.impl.ArbitrateConstants;
import com.alibaba.otter.shared.arbitrate.impl.setl.monitor.listener.NodeListener;
import com.alibaba.otter.shared.arbitrate.impl.zookeeper.ZooKeeperClient;
import com.alibaba.otter.shared.common.utils.zookeeper.ZkClientx;

/**
* otter所有node节点监控
*
* @author jianghang 2012-8-29 下午01:00:43
* @version 4.1.0
*/
public class NodeMonitor implements Monitor {

    private static final Logger logger     = LoggerFactory.getLogger(NodeMonitor.class);

    private ExecutorService     arbitrateExecutor;
    private ZkClientx           zookeeper  = ZooKeeperClient.getInstance();
    private List<NodeListener>  listeners  = Collections.synchronizedList(new ArrayList<NodeListener>());
    private volatile List<Long> aliveNodes = new ArrayList<Long>();                                      // se模块存活的节点
    private IZkChildListener    childListener;

    public NodeMonitor(){
        childListener = new IZkChildListener() {

            public void handleChildChange(String parentPath, List<String> currentChilds) throws Exception {
                if (currentChilds != null) {
                    initNodes(currentChilds);
                }
            }
        };
        List<String> childs = zookeeper.subscribeChildChanges(ArbitrateConstants.NODE_NID_ROOT, childListener);
        if (childs == null) {//如果为null,代表系统节点为初始化
            try {
                zookeeper.createPersistent(ArbitrateConstants.NODE_NID_ROOT, true);
            } catch (ZkNodeExistsException e) {
                //ignore
            }

            childs = zookeeper.getChildren(ArbitrateConstants.NODE_NID_ROOT);
        }

        initNodes(childs);
        // syncNodes();// 开始监视node节点的变化
        MonitorScheduler.register(this);
    }

    public void reload() {
        try {
            initNodes();// 更新数据
        } catch (Exception e) {
        }

    }

    public void destory() {
        listeners.clear();

        zookeeper.unsubscribeChildChanges(ArbitrateConstants.NODE_NID_ROOT, childListener);
        MonitorScheduler.unRegister(this);
    }

    /**
     * 返回当前存活的node列表
     */
    public List<Long> getAliveNodes(boolean reload) {
        if (reload) {
            initNodes();
        }

        return new ArrayList<Long>(aliveNodes);
    }

    /**
     * 返回当前存活的node列表
     */
    public List<Long> getAliveNodes() {
        return getAliveNodes(false);
    }

    private void initNodes() {
        // 获取一下当前存活的所有node节点,再根据对应pipeline关联的node,检查是否在对应的存活列表里
        List<String> nodes = zookeeper.getChildren(ArbitrateConstants.NODE_NID_ROOT);
        initNodes(nodes);
    }

    private synchronized void initNodes(List<String> nodes) {
        List<Long> nids = new ArrayList<Long>();
        for (String node : nodes) {
            if (StringUtils.isNumeric(node)) {
                nids.add(Long.valueOf(node));
            }
        }
        Collections.sort(nids);

        if (!aliveNodes.equals(nids)) {// 不相同,说明有变化
            if (logger.isDebugEnabled()) {
                logger.debug("old aliveNodes{} ,current aliveNodes{}", new Object[] { aliveNodes, nids });
            }

            aliveNodes = nids; // 切换引用,需设置为volatile保证线程安全&可见性
            processChanged(nids);// 通知变化
        }
    }

    // private void syncNodes() {
    // try {
    // List<String> nodes = zookeeper.getChildren(ArbitrateConstants.NODE_NID_ROOT, new AsyncWatcher() {
    //
    // public void asyncProcess(WatchedEvent event) {
    // // 出现session expired/connection losscase下,会触发所有的watcher响应,同时老的watcher会继续保留,所以会导致出现多次watcher响应
    // boolean dataChanged = event.getType() == EventType.NodeDataChanged
    // || event.getType() == EventType.NodeDeleted
    // || event.getType() == EventType.NodeCreated
    // || event.getType() == EventType.NodeChildrenChanged;
    // if (dataChanged) {
    // syncNodes();// 继续关注node节点变化
    // }
    // }
    // });
    //
    // initNodes(nodes);
    // } catch (KeeperException e) {
    // syncNodes();
    // logger.error("", e);
    // } catch (InterruptedException e) {
    // // ignore
    // }
    // }

    // ======================== listener处理 ======================

    public void addListener(NodeListener listener) {
        if (logger.isDebugEnabled()) {
            logger.debug("## pipeline[{}] add listener [{}]", ClassUtils.getShortClassName(listener.getClass()));
        }

        this.listeners.add(listener);
    }

    public void removeListener(NodeListener listener) {
        if (logger.isDebugEnabled()) {
            logger.debug("## remove listener [{}]", ClassUtils.getShortClassName(listener.getClass()));
        }

        this.listeners.remove(listener);
    }

    private void processChanged(final List<Long> nodes) {
        for (final NodeListener listener : listeners) {
            // 异步处理
            arbitrateExecutor.submit(new Runnable() {

                public void run() {
                    listener.processChanged(nodes);
                }
            });
        }
    }

    // ========= setter ========

    public void setArbitrateExecutor(ExecutorService arbitrateExecutor) {
        this.arbitrateExecutor = arbitrateExecutor;
    }

}
TOP

Related Classes of com.alibaba.otter.shared.arbitrate.impl.setl.monitor.NodeMonitor

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.