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

com.dangdang.config.service.zookeeper.ZookeeperConfigGroup Maven / Gradle / Ivy

/**
 * Copyright 1999-2014 dangdang.com.
 *  
 * 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.dangdang.config.service.zookeeper;

import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import javax.annotation.PreDestroy;

import org.apache.commons.lang3.tuple.ImmutablePair;
import org.apache.commons.lang3.tuple.Pair;
import org.apache.curator.framework.CuratorFramework;
import org.apache.curator.framework.CuratorFrameworkFactory;
import org.apache.curator.framework.api.CuratorListener;
import org.apache.curator.framework.api.GetChildrenBuilder;
import org.apache.curator.framework.api.GetDataBuilder;
import org.apache.curator.utils.ZKPaths;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.dangdang.config.service.ConfigGroup;
import com.dangdang.config.service.GeneralConfigGroup;
import com.google.common.base.Charsets;
import com.google.common.base.Throwables;
import com.google.common.collect.Maps;

/**
 * 配置组节点
 * 
 * @author Yuxuan Wang
 *
 */
public class ZookeeperConfigGroup extends GeneralConfigGroup {

	private static final long serialVersionUID = 1L;

	private ZookeeperConfigProfile configProfile;

	/**
	 * 节点名字
	 */
	private String node;

	private CuratorFramework client;

	private ConfigLocalCache configLocalCache;

	public void setConfigLocalCache(ConfigLocalCache configLocalCache) {
		this.configLocalCache = configLocalCache;
	}

	static final Logger LOGGER = LoggerFactory.getLogger(ZookeeperConfigGroup.class);

	public ZookeeperConfigGroup(ZookeeperConfigProfile configProfile, String node) {
		this(null, configProfile, node);
	}

	public ZookeeperConfigGroup(ConfigGroup internalConfigGroup, ZookeeperConfigProfile configProfile, String node) {
		super(internalConfigGroup);
		this.configProfile = configProfile;
		this.node = node;
		initConfigs();
	}

	private ScheduledExecutorService scheduler;

	private CuratorListener listener = new ConfigNodeEventListener(this);

	/**
	 * 初始化节点
	 */
	private void initConfigs() {
		client = CuratorFrameworkFactory.newClient(configProfile.getConnectStr(), configProfile.getRetryPolicy());
		client.start();
		client.getCuratorListenable().addListener(listener);

		LOGGER.debug("Loading properties for node: {}, with loading mode: {} and keys specified: {}", node, configProfile.getKeyLoadingMode(),
				configProfile.getKeysSpecified());
		loadNode();

		// Update local cache
		if (configLocalCache != null) {
			configLocalCache.saveLocalCache(this, node);
		}

		// Consistency check
		if (configProfile.isConsistencyCheck()) {
		        scheduler = Executors.newScheduledThreadPool(1);
			scheduler.scheduleAtFixedRate(new Runnable() {
				@Override
				public void run() {
					LOGGER.trace("Do consistency check for node: {}", node);
					loadNode();
				}
			}, 60000L, configProfile.getConsistencyCheckRate(), TimeUnit.MILLISECONDS);
		}
	}

	/**
	 * 加载节点并监听节点变化
	 */
	void loadNode() {
		final String nodePath = ZKPaths.makePath(configProfile.getVersionedRootNode(), node);

		GetChildrenBuilder childrenBuilder = client.getChildren();

		try {
			List children = childrenBuilder.watched().forPath(nodePath);
			if (children != null) {
				Map configs = Maps.newHashMap();
				for (String child : children) {
					Pair keyValue = loadKey(ZKPaths.makePath(nodePath, child));
					if (keyValue != null) {
						configs.put(keyValue.getKey(), keyValue.getValue());
					}
				}
				cleanAndputAll(configs);
			}
		} catch (Exception e) {
			throw Throwables.propagate(e);
		}
	}
	
	void reloadKey(final String nodePath) {
		try {
			Pair keyValue = loadKey(nodePath);
			if(keyValue != null) {
				super.put(keyValue.getKey(), keyValue.getValue());
			}
		} catch (Exception e) {
			throw Throwables.propagate(e);
		}
	}

	private Pair loadKey(final String nodePath) throws Exception {
		String nodeName = ZKPaths.getNodeFromPath(nodePath);
		Set keysSpecified = configProfile.getKeysSpecified();
		switch (configProfile.getKeyLoadingMode()) {
		case INCLUDE:
			if (keysSpecified == null || !keysSpecified.contains(nodeName)) {
				return null;
			}
			break;
		case EXCLUDE:
			if (keysSpecified.contains(nodeName)) {
				return null;
			}
			break;
		case ALL:
			break;
		default:
			break;
		}

		GetDataBuilder data = client.getData();
		String value = new String(data.watched().forPath(nodePath), Charsets.UTF_8);
		return new ImmutablePair(nodeName, value);
	}

	public String getNode() {
		return node;
	}

	public ConfigLocalCache getConfigLocalCache() {
		return configLocalCache;
	}

	/**
	 * 导出属性列表
	 * 
	 * @return
	 */
	public Map exportProperties() {
		return Maps.newHashMap(this);
	}

	@PreDestroy
	@Override
	public void close() {
		if (scheduler != null) {
			scheduler.shutdown();
		}
		if (client != null) {
			client.getCuratorListenable().removeListener(listener);
			client.close();
		}

	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy