com.alibaba.rocketmq.broker.longpolling.PullRequestHoldService Maven / Gradle / Ivy
The newest version!
/**
* Copyright (C) 2010-2013 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.rocketmq.broker.longpolling;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.alibaba.rocketmq.broker.BrokerController;
import com.alibaba.rocketmq.common.ServiceThread;
import com.alibaba.rocketmq.common.constant.LoggerName;
import com.alibaba.rocketmq.remoting.exception.RemotingCommandException;
/**
* 拉消息请求管理,如果拉不到消息,则在这里Hold住,等待消息到来
*
* @author shijia.wxr
* @since 2013-7-26
*/
public class PullRequestHoldService extends ServiceThread {
private static final Logger log = LoggerFactory.getLogger(LoggerName.BrokerLoggerName);
private static final String TOPIC_QUEUEID_SEPARATOR = "@";
private ConcurrentHashMap pullRequestTable =
new ConcurrentHashMap(1024);
private final BrokerController brokerController;
public PullRequestHoldService(final BrokerController brokerController) {
this.brokerController = brokerController;
}
private String buildKey(final String topic, final int queueId) {
StringBuilder sb = new StringBuilder();
sb.append(topic);
sb.append(TOPIC_QUEUEID_SEPARATOR);
sb.append(queueId);
return sb.toString();
}
public void suspendPullRequest(final String topic, final int queueId, final PullRequest pullRequest) {
String key = this.buildKey(topic, queueId);
ManyPullRequest mpr = this.pullRequestTable.get(key);
if (null == mpr) {
mpr = new ManyPullRequest();
ManyPullRequest prev = this.pullRequestTable.putIfAbsent(key, mpr);
if (prev != null) {
mpr = prev;
}
}
mpr.addPullRequest(pullRequest);
}
private void checkHoldRequest() {
for (String key : this.pullRequestTable.keySet()) {
String[] kArray = key.split(TOPIC_QUEUEID_SEPARATOR);
if (kArray != null && 2 == kArray.length) {
String topic = kArray[0];
int queueId = Integer.parseInt(kArray[1]);
final long offset =
this.brokerController.getMessageStore().getMaxOffsetInQuque(topic, queueId);
this.notifyMessageArriving(topic, queueId, offset);
}
}
}
public void notifyMessageArriving(final String topic, final int queueId, final long maxOffset) {
String key = this.buildKey(topic, queueId);
ManyPullRequest mpr = this.pullRequestTable.get(key);
if (mpr != null) {
List requestList = mpr.cloneListAndClear();
if (requestList != null) {
List replayList = new ArrayList();
for (PullRequest request : requestList) {
// 查看是否offset OK
if (maxOffset > request.getPullFromThisOffset()) {
try {
this.brokerController.getPullMessageProcessor().excuteRequestWhenWakeup(
request.getClientChannel(), request.getRequestCommand());
}
catch (RemotingCommandException e) {
log.error("", e);
}
continue;
}
// 尝试取最新Offset
else {
final long newestOffset =
this.brokerController.getMessageStore().getMaxOffsetInQuque(topic, queueId);
if (newestOffset > request.getPullFromThisOffset()) {
try {
this.brokerController.getPullMessageProcessor().excuteRequestWhenWakeup(
request.getClientChannel(), request.getRequestCommand());
}
catch (RemotingCommandException e) {
log.error("", e);
}
continue;
}
}
// 查看是否超时
if (System.currentTimeMillis() >= (request.getSuspendTimestamp() + request
.getTimeoutMillis())) {
try {
this.brokerController.getPullMessageProcessor().excuteRequestWhenWakeup(
request.getClientChannel(), request.getRequestCommand());
}
catch (RemotingCommandException e) {
log.error("", e);
}
continue;
}
// 当前不满足要求,重新放回Hold列表中
replayList.add(request);
}
if (!replayList.isEmpty()) {
mpr.addPullRequest(replayList);
}
}
}
}
@Override
public void run() {
log.info(this.getServiceName() + " service started");
while (!this.isStoped()) {
try {
this.waitForRunning(1000);
this.checkHoldRequest();
}
catch (Exception e) {
log.warn(this.getServiceName() + " service has exception. ", e);
}
}
log.info(this.getServiceName() + " service end");
}
@Override
public String getServiceName() {
return PullRequestHoldService.class.getSimpleName();
}
}