com.alibaba.otter.shared.arbitrate.impl.setl.monitor.NodeMonitor.java Source code

Java tutorial

Introduction

Here is the source code for com.alibaba.otter.shared.arbitrate.impl.setl.monitor.NodeMonitor.java

Source

/*
 * 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;

/**
 * otternode
 * 
 * @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;
    }

}