-
Notifications
You must be signed in to change notification settings - Fork 11.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Browse files
Browse the repository at this point in the history
- Loading branch information
Showing
30 changed files
with
3,227 additions
and
72 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
303 changes: 303 additions & 0 deletions
303
broker/src/main/java/org/apache/rocketmq/broker/pop/PopConsumerCache.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,303 @@ | ||
/* | ||
* Licensed to the Apache Software Foundation (ASF) under one or more | ||
* contributor license agreements. See the NOTICE file distributed with | ||
* this work for additional information regarding copyright ownership. | ||
* The ASF licenses this file to You 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 org.apache.rocketmq.broker.pop; | ||
|
||
import java.util.ArrayList; | ||
import java.util.Iterator; | ||
import java.util.List; | ||
import java.util.Map; | ||
import java.util.TreeMap; | ||
import java.util.concurrent.ConcurrentHashMap; | ||
import java.util.concurrent.ConcurrentMap; | ||
import java.util.concurrent.TimeUnit; | ||
import java.util.concurrent.atomic.AtomicInteger; | ||
import java.util.concurrent.locks.Lock; | ||
import java.util.concurrent.locks.ReentrantLock; | ||
import java.util.function.Consumer; | ||
import org.apache.rocketmq.broker.BrokerController; | ||
import org.apache.rocketmq.broker.offset.ConsumerOffsetManager; | ||
import org.apache.rocketmq.common.BrokerConfig; | ||
import org.apache.rocketmq.common.ServiceThread; | ||
import org.apache.rocketmq.common.constant.LoggerName; | ||
import org.apache.rocketmq.common.utils.ConcurrentHashMapUtils; | ||
import org.slf4j.Logger; | ||
import org.slf4j.LoggerFactory; | ||
|
||
public class PopConsumerCache extends ServiceThread { | ||
|
||
private static final Logger log = LoggerFactory.getLogger(LoggerName.ROCKETMQ_POP_LOGGER_NAME); | ||
private static final long OFFSET_NOT_EXIST = -1L; | ||
|
||
private final BrokerController brokerController; | ||
private final PopConsumerKVStore consumerRecordStore; | ||
private final PopConsumerLockService consumerLockService; | ||
private final Consumer<PopConsumerRecord> reviveConsumer; | ||
|
||
private final AtomicInteger estimateCacheSize; | ||
private final ConcurrentMap<String, ConsumerRecords> consumerRecordTable; | ||
|
||
public PopConsumerCache(BrokerController brokerController, PopConsumerKVStore consumerRecordStore, | ||
PopConsumerLockService popConsumerLockService, Consumer<PopConsumerRecord> reviveConsumer) { | ||
|
||
this.reviveConsumer = reviveConsumer; | ||
this.brokerController = brokerController; | ||
this.consumerRecordStore = consumerRecordStore; | ||
this.consumerLockService = popConsumerLockService; | ||
this.estimateCacheSize = new AtomicInteger(); | ||
this.consumerRecordTable = new ConcurrentHashMap<>(); | ||
} | ||
|
||
public String getKey(String groupId, String topicId, int queueId) { | ||
return groupId + "@" + topicId + "@" + queueId; | ||
} | ||
|
||
public String getKey(PopConsumerRecord consumerRecord) { | ||
return consumerRecord.getGroupId() + "@" + consumerRecord.getTopicId() + "@" + consumerRecord.getQueueId(); | ||
} | ||
|
||
public int getCacheKeySize() { | ||
return this.consumerRecordTable.size(); | ||
} | ||
|
||
public int getCacheSize() { | ||
return this.estimateCacheSize.intValue(); | ||
} | ||
|
||
public boolean isCacheFull() { | ||
return this.estimateCacheSize.intValue() > brokerController.getBrokerConfig().getPopCkMaxBufferSize(); | ||
} | ||
|
||
public long getMinOffsetInCache(String groupId, String topicId, int queueId) { | ||
ConsumerRecords consumerRecords = consumerRecordTable.get(this.getKey(groupId, topicId, queueId)); | ||
return consumerRecords != null ? consumerRecords.getMinOffsetInBuffer() : OFFSET_NOT_EXIST; | ||
} | ||
|
||
public long getPopInFlightMessageCount(String groupId, String topicId, int queueId) { | ||
ConsumerRecords consumerRecords = consumerRecordTable.get(this.getKey(groupId, topicId, queueId)); | ||
return consumerRecords != null ? consumerRecords.getInFlightRecordCount() : 0L; | ||
} | ||
|
||
public void writeRecords(List<PopConsumerRecord> consumerRecordList) { | ||
this.estimateCacheSize.addAndGet(consumerRecordList.size()); | ||
consumerRecordList.forEach(consumerRecord -> { | ||
ConsumerRecords consumerRecords = ConcurrentHashMapUtils.computeIfAbsent(consumerRecordTable, | ||
this.getKey(consumerRecord), k -> new ConsumerRecords(brokerController.getBrokerConfig(), | ||
consumerRecord.getGroupId(), consumerRecord.getTopicId(), consumerRecord.getQueueId())); | ||
assert consumerRecords != null; | ||
consumerRecords.write(consumerRecord); | ||
}); | ||
} | ||
|
||
/** | ||
* Remove the record from the input list then return the content that has not been deleted | ||
*/ | ||
public List<PopConsumerRecord> deleteRecords(List<PopConsumerRecord> consumerRecordList) { | ||
int total = consumerRecordList.size(); | ||
List<PopConsumerRecord> remain = new ArrayList<>(); | ||
consumerRecordList.forEach(consumerRecord -> { | ||
ConsumerRecords consumerRecords = consumerRecordTable.get(this.getKey(consumerRecord)); | ||
if (consumerRecords == null || !consumerRecords.delete(consumerRecord)) { | ||
remain.add(consumerRecord); | ||
} | ||
}); | ||
this.estimateCacheSize.addAndGet(remain.size() - total); | ||
return remain; | ||
} | ||
|
||
public int cleanupRecords(Consumer<PopConsumerRecord> consumer) { | ||
int remain = 0; | ||
Iterator<Map.Entry<String, ConsumerRecords>> iterator = consumerRecordTable.entrySet().iterator(); | ||
while (iterator.hasNext()) { | ||
// revive or write record to store | ||
ConsumerRecords records = iterator.next().getValue(); | ||
boolean timeout = consumerLockService.isLockTimeout( | ||
records.getGroupId(), records.getTopicId()); | ||
|
||
if (timeout) { | ||
List<PopConsumerRecord> removeExpiredRecords = | ||
records.removeExpiredRecords(Long.MAX_VALUE); | ||
if (removeExpiredRecords != null) { | ||
consumerRecordStore.writeRecords(removeExpiredRecords); | ||
} | ||
log.info("PopConsumerOffline, so clean expire records, groupId={}, topic={}, queueId={}, records={}", | ||
records.getGroupId(), records.getTopicId(), records.getQueueId(), | ||
removeExpiredRecords != null ? removeExpiredRecords.size() : 0); | ||
iterator.remove(); | ||
continue; | ||
} | ||
|
||
long currentTime = System.currentTimeMillis(); | ||
List<PopConsumerRecord> writeConsumerRecords = new ArrayList<>(); | ||
List<PopConsumerRecord> consumerRecords = records.removeExpiredRecords(currentTime); | ||
if (consumerRecords != null) { | ||
consumerRecords.forEach(consumerRecord -> { | ||
if (consumerRecord.getVisibilityTimeout() <= currentTime) { | ||
consumer.accept(consumerRecord); | ||
} else { | ||
writeConsumerRecords.add(consumerRecord); | ||
} | ||
}); | ||
} | ||
|
||
// write to store and handle it later | ||
consumerRecordStore.writeRecords(writeConsumerRecords); | ||
|
||
// commit min offset in buffer to offset store | ||
long offset = records.getMinOffsetInBuffer(); | ||
if (offset > OFFSET_NOT_EXIST) { | ||
this.commitOffset("PopConsumerCache", | ||
records.getGroupId(), records.getTopicId(), records.getQueueId(), offset); | ||
} | ||
|
||
remain += records.getInFlightRecordCount(); | ||
} | ||
return remain; | ||
} | ||
|
||
public void commitOffset(String clientHost, String groupId, String topicId, int queueId, long offset) { | ||
if (!consumerLockService.tryLock(groupId, topicId)) { | ||
return; | ||
} | ||
try { | ||
ConsumerOffsetManager consumerOffsetManager = brokerController.getConsumerOffsetManager(); | ||
long commit = consumerOffsetManager.queryOffset(groupId, topicId, queueId); | ||
if (commit != OFFSET_NOT_EXIST && offset < commit) { | ||
log.info("PopConsumerCache, consumer offset less than store, " + | ||
"groupId={}, topicId={}, queueId={}, offset={}", groupId, topicId, queueId, offset); | ||
} | ||
consumerOffsetManager.commitOffset(clientHost, groupId, topicId, queueId, offset); | ||
} finally { | ||
consumerLockService.unlock(groupId, topicId); | ||
} | ||
} | ||
|
||
public void removeRecords(String groupId, String topicId, int queueId) { | ||
this.consumerRecordTable.remove(this.getKey(groupId, topicId, queueId)); | ||
} | ||
|
||
@Override | ||
public String getServiceName() { | ||
return PopConsumerCache.class.getSimpleName(); | ||
} | ||
|
||
@Override | ||
public void run() { | ||
while (!this.isStopped()) { | ||
try { | ||
this.waitForRunning(TimeUnit.SECONDS.toMillis(1)); | ||
int cacheSize = this.cleanupRecords(reviveConsumer); | ||
this.estimateCacheSize.set(cacheSize); | ||
} catch (Exception e) { | ||
log.error("PopConsumerCacheService revive error", e); | ||
} | ||
} | ||
} | ||
|
||
protected static class ConsumerRecords { | ||
|
||
private final Lock lock; | ||
private final String groupId; | ||
private final String topicId; | ||
private final int queueId; | ||
private final BrokerConfig brokerConfig; | ||
private final TreeMap<Long /* offset */, PopConsumerRecord> recordTreeMap; | ||
|
||
public ConsumerRecords(BrokerConfig brokerConfig, String groupId, String topicId, int queueId) { | ||
this.groupId = groupId; | ||
this.topicId = topicId; | ||
this.queueId = queueId; | ||
this.lock = new ReentrantLock(); | ||
this.brokerConfig = brokerConfig; | ||
this.recordTreeMap = new TreeMap<>(); | ||
} | ||
|
||
public void write(PopConsumerRecord record) { | ||
lock.lock(); | ||
try { | ||
recordTreeMap.put(record.getOffset(), record); | ||
} finally { | ||
lock.unlock(); | ||
} | ||
} | ||
|
||
public boolean delete(PopConsumerRecord record) { | ||
PopConsumerRecord popConsumerRecord; | ||
lock.lock(); | ||
try { | ||
popConsumerRecord = recordTreeMap.remove(record.getOffset()); | ||
} finally { | ||
lock.unlock(); | ||
} | ||
return popConsumerRecord != null; | ||
} | ||
|
||
public long getMinOffsetInBuffer() { | ||
Map.Entry<Long, PopConsumerRecord> entry = recordTreeMap.firstEntry(); | ||
return entry != null ? entry.getKey() : OFFSET_NOT_EXIST; | ||
} | ||
|
||
public int getInFlightRecordCount() { | ||
return recordTreeMap.size(); | ||
} | ||
|
||
public List<PopConsumerRecord> removeExpiredRecords(long currentTime) { | ||
List<PopConsumerRecord> result = null; | ||
lock.lock(); | ||
try { | ||
Iterator<Map.Entry<Long, PopConsumerRecord>> iterator = recordTreeMap.entrySet().iterator(); | ||
while (iterator.hasNext()) { | ||
Map.Entry<Long, PopConsumerRecord> entry = iterator.next(); | ||
// org.apache.rocketmq.broker.processor.PopBufferMergeService.scan | ||
if (entry.getValue().getVisibilityTimeout() <= currentTime || | ||
entry.getValue().getPopTime() + brokerConfig.getPopCkStayBufferTime() <= currentTime) { | ||
if (result == null) { | ||
result = new ArrayList<>(); | ||
} | ||
result.add(entry.getValue()); | ||
iterator.remove(); | ||
} | ||
} | ||
} finally { | ||
lock.unlock(); | ||
} | ||
return result; | ||
} | ||
|
||
public String getGroupId() { | ||
return groupId; | ||
} | ||
|
||
public String getTopicId() { | ||
return topicId; | ||
} | ||
|
||
public int getQueueId() { | ||
return queueId; | ||
} | ||
|
||
@Override | ||
public String toString() { | ||
return "ConsumerRecords{" + | ||
"lock=" + lock + | ||
", topicId=" + topicId + | ||
", groupId=" + groupId + | ||
", queueId=" + queueId + | ||
", recordTreeMap=" + recordTreeMap.size() + | ||
'}'; | ||
} | ||
} | ||
} |
Oops, something went wrong.