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

qunar.tc.qmq.backup.service.impl.DeadMessageContentBatchBackup Maven / Gradle / Ivy

There is a newer version: 1.1.43
Show newest version
/*
 * Copyright 2018 Qunar, Inc.
 *
 * 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 qunar.tc.qmq.backup.service.impl;

import static qunar.tc.qmq.backup.config.DefaultBackupConfig.DEFAULT_BATCH_SIZE;
import static qunar.tc.qmq.backup.config.DefaultBackupConfig.DEFAULT_RETRY_NUM;
import static qunar.tc.qmq.backup.config.DefaultBackupConfig.MESSAGE_BATCH_SIZE_CONFIG_KEY;
import static qunar.tc.qmq.backup.config.DefaultBackupConfig.MESSAGE_RETRY_NUM_CONFIG_KEY;
import static qunar.tc.qmq.metrics.MetricsConstants.EMPTY;
import static qunar.tc.qmq.metrics.MetricsConstants.SUBJECT_ARRAY;
import static qunar.tc.qmq.utils.RetrySubjectUtils.getConsumerGroup;

import java.util.Date;
import java.util.List;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.TimeUnit;
import java.util.function.Consumer;

import com.google.common.base.Throwables;
import org.jboss.netty.util.CharsetUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import qunar.tc.qmq.backup.base.BackupQuery;
import qunar.tc.qmq.backup.config.BackupConfig;
import qunar.tc.qmq.backup.service.BackupKeyGenerator;
import qunar.tc.qmq.backup.service.MessageService;
import qunar.tc.qmq.backup.store.KvStore;
import qunar.tc.qmq.metrics.Metrics;
import qunar.tc.qmq.store.MessageQueryIndex;
import qunar.tc.qmq.utils.RetrySubjectUtils;

/**
 * @author xufeng.deng [email protected]
 * @since 2018-12-10 16:53
 */
public class DeadMessageContentBatchBackup extends AbstractBatchBackup {
    private static final Logger LOGGER = LoggerFactory.getLogger(DeadMessageContentBatchBackup.class);

    private final KvStore deadMessageStore;
    private final BackupKeyGenerator keyGenerator;
    private final MessageService messageService;


    public DeadMessageContentBatchBackup(KvStore deadMessageStore, BackupKeyGenerator keyGenerator, BackupConfig config, MessageService messageService) {
        super("deadMessageContentBackup", config);
        this.deadMessageStore = deadMessageStore;
        this.keyGenerator = keyGenerator;
        this.messageService = messageService;
    }

    @Override
    protected void doStop() {
    }

    @Override
    protected void store(List batch, Consumer fi) {
        doStore(batch, fi);
    }

	private void doStore(List batch, Consumer func) {
		for (int i = 0; i < retryNum(); ++i) {
			try {
				doBatchSaveBackupDeadMessage(batch, func);
				return;
			}
			catch (Exception e) {
				LOGGER.error("dead message backup store error.", e);
				monitorStoreRetry();
			}
		}

		monitorStoreDiscard();
	}

    private void doBatchSaveBackupDeadMessage(List indexes, Consumer func) {
        byte[][] recordKeys = new byte[indexes.size()][];
        byte[][][] recordValues = new byte[indexes.size()][][];
        long currentTime = System.currentTimeMillis();
        try {
            MessageQueryIndex tailIndex = null;
            for (int i = 0; i < indexes.size(); ++i) {
                MessageQueryIndex index = indexes.get(i);
                String subject = index.getSubject();

                BackupQuery query = new BackupQuery();
                query.setSubject(subject);
                query.setBrokerGroup(new String(brokerGroupBytes, CharsetUtil.UTF_8));
                query.setSequence(index.getSequence());

                CompletableFuture future = messageService.findMessageBytes(query);


                try {
                    monitorDeadMessageQps(subject);
                    LOGGER.info("backup dead msg content, subject: {}", subject);
                    final String realSubject = RetrySubjectUtils.getRealSubject(subject);
                    final String messageId = index.getMessageId();
                    final String consumerGroup = getConsumerGroup(subject);
                    final Date createTime = new Date(index.getCreateTime());
                    final byte[] key = keyGenerator.generateDeadMessageKey(realSubject, messageId, consumerGroup, createTime);

                    byte[] value = future.get(1, TimeUnit.SECONDS);


                    final byte[][] recordValue = new byte[][]{value};
                    recordKeys[i] = key;
                    recordValues[i] = recordValue;

                    if (tailIndex == null || tailIndex.compareTo(index) < 0) tailIndex = index;
                } catch (Exception e) {
                    LOGGER.error("batch backup dead message failed.", e);
                    monitorDeadMessageError(subject);
                }
            }

            deadMessageStore.batchSave(recordKeys, recordValues);
            if (func != null) func.accept(tailIndex);
        } catch (Throwable e) {
            LOGGER.error("put backup dead message fail.", e);
            Throwables.propagate(e);
        } finally {
            Metrics.timer("BatchBackup.Store.Timer", TYPE_ARRAY, DEAD_MESSAGE_CONTENT_TYPE).update(System.currentTimeMillis() - currentTime, TimeUnit.MILLISECONDS);
        }
    }

    private int retryNum() {
        return config.getInt(MESSAGE_RETRY_NUM_CONFIG_KEY, DEFAULT_RETRY_NUM);
    }

    private static void monitorDeadMessageError(String subject) {
        Metrics.counter("dead_message_content_store_error", SUBJECT_ARRAY, new String[]{subject}).inc();
    }

	private static void monitorStoreDiscard() {
		Metrics.counter("dead_message_content_backup_store_discard", EMPTY, EMPTY).inc();
	}

    private static void monitorDeadMessageQps(String subject) {
        Metrics.meter("backup.dead.message.content.qps", SUBJECT_ARRAY, new String[]{subject}).mark();
    }

    private static void monitorStoreRetry() {
		Metrics.counter("dead_message_content_backup_store_retry", EMPTY, EMPTY).inc();
    }

    @Override
    protected int getBatchSize() {
        return config.getInt(MESSAGE_BATCH_SIZE_CONFIG_KEY, DEFAULT_BATCH_SIZE);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy