com.taobao.metamorphosis.metaslave.SubscribeHandler Maven / Gradle / Ivy
/*
* (C) 2007-2012 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.
* Authors:
* wuhua
*/
package com.taobao.metamorphosis.metaslave;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.UnsupportedEncodingException;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.atomic.AtomicBoolean;
import org.apache.commons.lang.StringUtils;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import com.taobao.gecko.core.command.ResponseStatus;
import com.taobao.gecko.core.util.OpaqueGenerator;
import com.taobao.gecko.service.exception.NotifyRemotingException;
import com.taobao.metamorphosis.client.MetaClientConfig;
import com.taobao.metamorphosis.client.consumer.ConsumerConfig;
import com.taobao.metamorphosis.client.consumer.ConsumerZooKeeper.ZKLoadRebalanceListener;
import com.taobao.metamorphosis.client.consumer.FetchManager;
import com.taobao.metamorphosis.client.consumer.MessageConsumer;
import com.taobao.metamorphosis.client.consumer.MessageListener;
import com.taobao.metamorphosis.client.consumer.SimpleMessageConsumer;
import com.taobao.metamorphosis.cluster.Partition;
import com.taobao.metamorphosis.exception.MetaClientException;
import com.taobao.metamorphosis.network.BooleanCommand;
import com.taobao.metamorphosis.network.StatsCommand;
import com.taobao.metamorphosis.server.assembly.MetaMorphosisBroker;
import com.taobao.metamorphosis.server.store.MessageStore;
import com.taobao.metamorphosis.server.utils.MetaConfig;
import com.taobao.metamorphosis.server.utils.MetaMBeanServer;
/**
* ?????master??????Ϣ???洢??slaver
*
* @author ??,dennis
* @since 2011-6-24 ????06:03:29
*/
public class SubscribeHandler implements SubscribeHandlerMBean {
private static final String SLAVE_CONFIG_ENCODING = System.getProperty("meta.slave.config.encoding", "GBK");
private final static Log log = LogFactory.getLog(SubscribeHandler.class);
private final MetaMorphosisBroker broker;
private final SlaveZooKeeper slaveZooKeeper;
private final SlaveMetaMessageSessionFactory sessionFactory;
private final MessageListener messageListener;
private MessageConsumer consumer;
private final SlaveOffsetStorage slaveOffsetStorage;
private final AtomicBoolean isStarted = new AtomicBoolean(false);
private String masterServerUrl;
public SubscribeHandler(final MetaMorphosisBroker broker) throws MetaClientException {
this.broker = broker;
this.slaveZooKeeper = new SlaveZooKeeper(this.broker, this);
final MetaConfig metaConfig = this.broker.getMetaConfig();
final MetaClientConfig metaClientConfig = new MetaClientConfig();
metaClientConfig.setZkConfig(metaConfig.getZkConfig());
this.sessionFactory = SlaveMetaMessageSessionFactory.create(metaClientConfig, metaConfig.getBrokerId());
this.slaveOffsetStorage =
new SlaveOffsetStorage(this.broker, this.slaveZooKeeper, this.sessionFactory.getRemotingClient());
this.messageListener =
new MetaSlaveListener(this.broker.getBrokerZooKeeper(), this.broker.getStoreManager(),
new SlaveStatsManager(this.broker.getStatsManager()));
MetaMBeanServer.registMBean(this, null);
}
synchronized public void tryReloadConfig(long masterChecksum) throws IOException {
MetaConfig metaConfig = this.broker.getMetaConfig();
if (metaConfig == null || metaConfig.getSlaveConfig() == null) {
return;
}
if (!metaConfig.getSlaveConfig().isAutoSyncMasterConfig()) {
return;
}
// Master Config file changed
if (metaConfig.getConfigFileChecksum() != masterChecksum) {
String masterUrl = this.slaveZooKeeper.getMasterServerUrl();
for (int i = 0; i < 3; i++) {
try {
BooleanCommand resp =
(BooleanCommand) this.sessionFactory.getRemotingClient().invokeToGroup(masterUrl,
new StatsCommand(OpaqueGenerator.getNextOpaque(), "config"));
if (resp.getResponseStatus() == ResponseStatus.NO_ERROR) {
this.tryReloadConfig(resp);
break;
}
else {
log.error("Get config file failed,retry " + (i + 1) + " times,error code:" + resp.getCode());
}
}
catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
catch (Exception e) {
log.error("Stats new config file from master failed,retry " + (i + 1) + " times", e);
}
}
}
}
private void tryReloadConfig(BooleanCommand resp) throws UnsupportedEncodingException, IOException,
FileNotFoundException {
String str = resp.getErrorMsg();
str = new String(str.getBytes("utf-8"));
MetaConfig newConfig = new MetaConfig();
newConfig.loadFromString(str);
// If topics config changed
if (!newConfig.getTopicConfigMap().equals(this.broker.getMetaConfig().getTopicConfigMap())) {
File tmpFile = File.createTempFile("meta_config", "slave_sync");
BufferedWriter writer =
new BufferedWriter(new OutputStreamWriter(new FileOutputStream(tmpFile), SLAVE_CONFIG_ENCODING));
writer.write(str);
writer.flush();
writer.close();
if (!tmpFile.renameTo(new File(this.broker.getMetaConfig().getConfigFilePath()))) {
log.error("Write new config file failed");
}
else {
log.info("Write new config file from master to slave local");
log.info("Trying to reload the new config file from master...");
this.broker.getMetaConfig().reload();
}
}
}
// ??????slave broker????֮??
synchronized public void start() {
if (this.isStarted.get()) {
log.info("Subscriber has been started");
return;
}
try {
log.info("Try to check if master config file changed...");
this.tryReloadConfig(-1L);
final Map> partitionsForTopics =
this.slaveZooKeeper.getPartitionsForTopicsFromMaster();
if (partitionsForTopics == null || partitionsForTopics.isEmpty()) {
throw new SubscribeMasterMessageException("topics in master is empty");
}
// ??????Ϣ?ļ?,????б?Ҫ?Ļ?
this.handleMassageFiles(partitionsForTopics);
this.initMessageConsumer();
this.subscribeMetaMaster(partitionsForTopics.keySet());
this.isStarted.set(true);
log.info("start subscribe message from master success");
}
catch (final Throwable e) {
// ????masterû??????????ԭ???????????ݸ???ʧ?ܵ?,??¼һ??,
// ??????????zk,????masterʱ???????ݸ???
log.warn("problem occured in start subscribe message from master,maybe master not started or other errers",
e);
log.info("waiting for master start next time...");
}
this.slaveZooKeeper.start();
}
/** ??һ??topic??ij??patition????????Ϣ?ļ?ʱ,??master??ѯoffset?????????offset????store???ļ? **/
private void handleMassageFiles(final Map> partitionsForTopics) {
try {
this.masterServerUrl = this.slaveZooKeeper.getMasterServerUrl();
for (final Map.Entry> each : partitionsForTopics.entrySet()) {
for (final Partition partition : each.getValue()) {
final String topic = each.getKey();
final MessageStore messageStore =
this.broker.getStoreManager().getMessageStore(topic, partition.getPartition());
// ???ز???????Ϣ?ļ?
final StringBuilder logStr = new StringBuilder();
if (messageStore == null) {
logStr.append("Local file for topic=").append(topic).append(" is not exist,partition=")
.append(partition.getPartition()).append("\n");
final long offsetInMaster =
this.slaveOffsetStorage.queryOffsetInMaster(this.masterServerUrl, partition, topic);
logStr.append("query offset from master,offset=").append(offsetInMaster).append("\n");
// ????һ????ָ??offset??ʼ??store???ļ?
if (offsetInMaster > 0) {
this.broker.getStoreManager().getOrCreateMessageStore(topic, partition.getPartition(),
offsetInMaster);
logStr.append("create messageStore,offset=").append(offsetInMaster).append("\n");
}
log.info(logStr.toString());
}
}
}
}
catch (final Throwable e) {
throw new SubscribeMasterMessageException("errer occur on handleMassageFiles", e);
}
}
public void closeConnectIfNeed() {
if (StringUtils.isBlank(this.masterServerUrl)) {
log.warn("can not close connect,master server url is blank");
return;
}
try {
this.sessionFactory.getRemotingClient().close(this.masterServerUrl, false);
log.info("Closing " + this.masterServerUrl);
if (log.isDebugEnabled()) {
log.debug("connect count="
+ this.sessionFactory.getRemotingClient().getConnectionCount(this.masterServerUrl));
}
}
catch (final NotifyRemotingException e) {
log.error("close connect " + this.masterServerUrl + " failed", e);
}
}
public void stop() {
try {
if (this.isStarted.compareAndSet(true, false)) {
log.info("stop subscribe from master");
this.consumer.shutdown();
this.consumer = null;
log.info("stop success");
}
else {
log.info("can not stop,since this not started");
}
}
catch (final MetaClientException e) {
log.error("stop consumer failed", e);
}
}
public void shutdown() {
this.stop();
try {
this.sessionFactory.shutdown();
}
catch (final MetaClientException e) {
log.error(e);
}
}
private void initMessageConsumer() throws Exception {
if (this.consumer == null) {
final ConsumerConfig consumerConfig =
new ConsumerConfig(this.broker.getMetaConfig().getSlaveConfig().getSlaveGroup());
consumerConfig.setMaxDelayFetchTimeInMills(this.broker.getMetaConfig().getSlaveConfig()
.getSlaveMaxDelayInMills());
consumerConfig.setMaxFetchRetries(Integer.MAX_VALUE);// ??Ϣ????ʧ?ܿ?????һ????,??????recover
this.consumer = this.sessionFactory.createConsumer(consumerConfig, this.slaveOffsetStorage);
}
else {
log.warn("consumer existed");
}
}
private void subscribeMetaMaster(final Set topics) throws Exception {
try {
for (final String topic : topics) {
this.consumer.subscribe(topic, 1024 * 1024, this.messageListener);
}
this.consumer.completeSubscribe();
}
catch (final Exception e) {
log.warn("start subscribe from meta master fail", e);
throw e;
}
}
@Override
public String getStatus() {
StringBuilder sb = new StringBuilder();
this.appendKeyValue(sb, "Subscriber handler", this.isStarted.get() ? "running" : "stop");
if (!this.isStarted() || this.consumer == null) {
return sb.toString();
}
SlaveConsumerZooKeeper scz = (SlaveConsumerZooKeeper) this.sessionFactory.getConsumerZooKeeper();
FetchManager fetchManager = ((SimpleMessageConsumer) this.consumer).getFetchManager();
ZKLoadRebalanceListener listener = scz.getBrokerConnectionListener(fetchManager);
Map> topicPartitions = listener.getTopicPartitions();
int totalPartitions = 0;
for (Set set : topicPartitions.values()) {
totalPartitions += set.size();
}
this.appendKeyValue(sb, "Replicate partitions", totalPartitions);
this.appendKeyValue(sb, "Replicate topic partitions detail", "");
for (Map.Entry> entry : topicPartitions.entrySet()) {
this.appendKeyValue(sb, " " + entry.getKey(), entry.getValue().size());
}
int fetchRequestCount = fetchManager.getFetchRequestCount();
this.appendKeyValue(sb, "Replicate requests", fetchRequestCount);
this.appendKeyValue(sb, "Replicate runner", fetchManager.isShutdown() ? "stop" : "running");
this.appendKeyValue(sb, "Replicate status",
fetchRequestCount == totalPartitions && !fetchManager.isShutdown() ? "yes" : "no");
return sb.toString();
}
private void appendKeyValue(StringBuilder sb, Object key, Object v) {
sb.append(key).append(":").append(v).append("\r\n");
}
// for test
SlaveZooKeeper getSlaveZooKeeper() {
return this.slaveZooKeeper;
}
// for test
boolean isStarted() {
return this.isStarted.get();
}
/*
* (non-Javadoc)
*
* @see com.taobao.metamorphosis.metaslave.SubscribeHandlerMBean#restart()
*/
@Override
public void restart() {
this.stop();
this.start();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy