Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
Apache RocketMQ
Rocketmq
提交
f0ad359c
R
Rocketmq
项目概览
Apache RocketMQ
/
Rocketmq
上一次同步 大约 3 年
通知
268
Star
16139
Fork
68
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
R
Rocketmq
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
提交
f0ad359c
编写于
6月 03, 2019
作者:
C
chengxiangwang
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
add logic of resending msg when acktimeout
上级
caf14397
变更
10
显示空白变更内容
内联
并排
Showing
10 changed file
with
158 addition
and
56 deletion
+158
-56
common/src/main/java/org/apache/rocketmq/common/MqttConfig.java
.../src/main/java/org/apache/rocketmq/common/MqttConfig.java
+10
-0
common/src/main/java/org/apache/rocketmq/common/service/NnodeService.java
...java/org/apache/rocketmq/common/service/NnodeService.java
+1
-1
mqtt/src/main/java/org/apache/rocketmq/mqtt/client/IOTClientManagerImpl.java
...org/apache/rocketmq/mqtt/client/IOTClientManagerImpl.java
+8
-2
mqtt/src/main/java/org/apache/rocketmq/mqtt/client/InFlightMessage.java
...java/org/apache/rocketmq/mqtt/client/InFlightMessage.java
+1
-8
mqtt/src/main/java/org/apache/rocketmq/mqtt/client/InFlightPacket.java
.../java/org/apache/rocketmq/mqtt/client/InFlightPacket.java
+89
-0
mqtt/src/main/java/org/apache/rocketmq/mqtt/client/MQTTSession.java
...ain/java/org/apache/rocketmq/mqtt/client/MQTTSession.java
+7
-42
mqtt/src/main/java/org/apache/rocketmq/mqtt/constant/MqttConstant.java
.../java/org/apache/rocketmq/mqtt/constant/MqttConstant.java
+1
-0
mqtt/src/main/java/org/apache/rocketmq/mqtt/service/impl/MqttScheduledServiceImpl.java
.../rocketmq/mqtt/service/impl/MqttScheduledServiceImpl.java
+38
-0
mqtt/src/main/java/org/apache/rocketmq/mqtt/task/MqttPushTask.java
...main/java/org/apache/rocketmq/mqtt/task/MqttPushTask.java
+2
-2
mqtt/src/test/java/org/apache/rocketmq/mqtt/MqttPubackMessageHandlerTest.java
...rg/apache/rocketmq/mqtt/MqttPubackMessageHandlerTest.java
+1
-1
未找到文件。
common/src/main/java/org/apache/rocketmq/common/MqttConfig.java
浏览文件 @
f0ad359c
...
...
@@ -54,6 +54,8 @@ public class MqttConfig {
private
long
persistOffsetInterval
=
2
*
1000
;
private
long
scanAckTimeoutInterval
=
1000
;
public
int
getListenPort
()
{
return
listenPort
;
}
...
...
@@ -149,4 +151,12 @@ public class MqttConfig {
public
void
setPersistOffsetInterval
(
long
persistOffsetInterval
)
{
this
.
persistOffsetInterval
=
persistOffsetInterval
;
}
public
long
getScanAckTimeoutInterval
()
{
return
scanAckTimeoutInterval
;
}
public
void
setScanAckTimeoutInterval
(
long
scanAckTimeoutInterval
)
{
this
.
scanAckTimeoutInterval
=
scanAckTimeoutInterval
;
}
}
common/src/main/java/org/apache/rocketmq/common/service/NnodeService.java
浏览文件 @
f0ad359c
mqtt/src/main/java/org/apache/rocketmq/mqtt/client/IOTClientManagerImpl.java
浏览文件 @
f0ad359c
...
...
@@ -23,6 +23,7 @@ import java.util.Map;
import
java.util.Set
;
import
java.util.TreeMap
;
import
java.util.concurrent.ConcurrentHashMap
;
import
java.util.concurrent.DelayQueue
;
import
org.apache.rocketmq.common.client.Client
;
import
org.apache.rocketmq.common.client.ClientManagerImpl
;
import
org.apache.rocketmq.common.client.Subscription
;
...
...
@@ -43,8 +44,9 @@ public class IOTClientManagerImpl extends ClientManagerImpl {
1024
);
private
final
ConcurrentHashMap
<
String
/*clientId*/
,
Subscription
>
clientId2Subscription
=
new
ConcurrentHashMap
<>(
1024
);
private
final
Map
<
String
/*snode ip*/
,
MqttClient
>
snode2MqttClient
=
new
HashMap
<>();
private
final
ConcurrentHashMap
<
String
/*broker*/
,
ConcurrentHashMap
<
String
/*topic@clientId*/
,
TreeMap
<
Long
/*queueOffset*/
,
MessageExt
>>>
processTable
=
new
ConcurrentHashMap
<>();
private
final
ConcurrentHashMap
<
String
/*broker*/
,
ConcurrentHashMap
<
String
/*rootTopic@clientId*/
,
TreeMap
<
Long
/*queueOffset*/
,
MessageExt
>>>
processTable
=
new
ConcurrentHashMap
<>();
private
final
ConcurrentHashMap
<
String
/*rootTopic@clientId*/
,
Integer
>
consumeOffsetTable
=
new
ConcurrentHashMap
<>();
private
final
DelayQueue
<
InFlightPacket
>
inflightTimeouts
=
new
DelayQueue
<>();
public
IOTClientManagerImpl
()
{
}
...
...
@@ -129,4 +131,8 @@ public class IOTClientManagerImpl extends ClientManagerImpl {
public
ConcurrentHashMap
<
String
,
ConcurrentHashMap
<
String
,
TreeMap
<
Long
,
MessageExt
>>>
getProcessTable
()
{
return
processTable
;
}
public
DelayQueue
<
InFlightPacket
>
getInflightTimeouts
()
{
return
inflightTimeouts
;
}
}
mqtt/src/main/java/org/apache/rocketmq/mqtt/client/InFlightMessage.java
浏览文件 @
f0ad359c
...
...
@@ -23,16 +23,13 @@ public class InFlightMessage {
private
final
Integer
pushQos
;
private
final
BrokerData
brokerData
;
private
final
byte
[]
body
;
private
final
String
messageId
;
private
final
long
queueOffset
;
public
InFlightMessage
(
String
topic
,
Integer
pushQos
,
byte
[]
body
,
BrokerData
brokerData
,
String
messageId
,
long
queueOffset
)
{
public
InFlightMessage
(
String
topic
,
Integer
pushQos
,
byte
[]
body
,
BrokerData
brokerData
,
long
queueOffset
)
{
this
.
topic
=
topic
;
this
.
pushQos
=
pushQos
;
this
.
body
=
body
;
this
.
brokerData
=
brokerData
;
this
.
messageId
=
messageId
;
this
.
queueOffset
=
queueOffset
;
}
...
...
@@ -44,10 +41,6 @@ public class InFlightMessage {
return
brokerData
;
}
public
String
getMessageId
()
{
return
messageId
;
}
public
long
getQueueOffset
()
{
return
queueOffset
;
}
...
...
mqtt/src/main/java/org/apache/rocketmq/mqtt/client/InFlightPacket.java
0 → 100644
浏览文件 @
f0ad359c
/*
* 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.mqtt.client
;
import
java.util.concurrent.Delayed
;
import
java.util.concurrent.TimeUnit
;
public
class
InFlightPacket
implements
Delayed
{
private
final
MQTTSession
client
;
private
final
int
packetId
;
private
long
startTime
;
private
int
resendTime
=
0
;
InFlightPacket
(
MQTTSession
client
,
int
packetId
,
long
delayInMilliseconds
)
{
this
.
client
=
client
;
this
.
packetId
=
packetId
;
this
.
startTime
=
System
.
currentTimeMillis
()
+
delayInMilliseconds
;
}
@Override
public
long
getDelay
(
TimeUnit
unit
)
{
long
diff
=
startTime
-
System
.
currentTimeMillis
();
return
unit
.
convert
(
diff
,
TimeUnit
.
MILLISECONDS
);
}
@Override
public
int
compareTo
(
Delayed
o
)
{
if
((
this
.
startTime
-
((
InFlightPacket
)
o
).
startTime
)
==
0
)
{
return
0
;
}
if
((
this
.
startTime
-
((
InFlightPacket
)
o
).
startTime
)
>
0
)
{
return
1
;
}
else
{
return
-
1
;
}
}
public
MQTTSession
getClient
()
{
return
client
;
}
public
int
getPacketId
()
{
return
packetId
;
}
public
long
getStartTime
()
{
return
startTime
;
}
public
void
setStartTime
(
long
startTime
)
{
this
.
startTime
=
startTime
;
}
public
int
getResendTime
()
{
return
resendTime
;
}
public
void
setResendTime
(
int
resendTime
)
{
this
.
resendTime
=
resendTime
;
}
@Override
public
boolean
equals
(
Object
obj
)
{
if
(
obj
==
this
)
{
return
true
;
}
if
(!(
obj
instanceof
InFlightPacket
))
{
return
false
;
}
InFlightPacket
packet
=
(
InFlightPacket
)
obj
;
return
packet
.
getClient
().
equals
(
this
.
getClient
())
&&
packet
.
getPacketId
()
==
this
.
getPacketId
();
}
}
\ No newline at end of file
mqtt/src/main/java/org/apache/rocketmq/mqtt/client/MQTTSession.java
浏览文件 @
f0ad359c
...
...
@@ -23,9 +23,6 @@ import java.util.Objects;
import
java.util.Set
;
import
java.util.TreeMap
;
import
java.util.concurrent.ConcurrentHashMap
;
import
java.util.concurrent.DelayQueue
;
import
java.util.concurrent.Delayed
;
import
java.util.concurrent.TimeUnit
;
import
java.util.concurrent.atomic.AtomicInteger
;
import
org.apache.rocketmq.common.client.Client
;
import
org.apache.rocketmq.common.client.ClientRole
;
...
...
@@ -45,6 +42,7 @@ import org.apache.rocketmq.remoting.netty.NettyChannelImpl;
import
org.apache.rocketmq.remoting.protocol.RemotingCommand
;
import
org.apache.rocketmq.remoting.transport.mqtt.MqttHeader
;
import
static
org
.
apache
.
rocketmq
.
mqtt
.
constant
.
MqttConstant
.
FLIGHT_BEFORE_RESEND_MS
;
import
static
org
.
apache
.
rocketmq
.
mqtt
.
constant
.
MqttConstant
.
TOPIC_CLIENTID_SEPARATOR
;
public
class
MQTTSession
extends
Client
{
...
...
@@ -57,40 +55,9 @@ public class MQTTSession extends Client {
private
final
DefaultMqttMessageProcessor
defaultMqttMessageProcessor
;
private
final
AtomicInteger
inflightSlots
=
new
AtomicInteger
(
10
);
private
final
Map
<
Integer
,
InFlightMessage
>
inflightWindow
=
new
HashMap
<>();
private
final
DelayQueue
<
InFlightPacket
>
inflightTimeouts
=
new
DelayQueue
<>();
private
static
final
int
FLIGHT_BEFORE_RESEND_MS
=
5_000
;
private
Hashtable
inUsePacketIds
=
new
Hashtable
();
private
int
nextPacketId
=
0
;
static
class
InFlightPacket
implements
Delayed
{
final
int
packetId
;
private
long
startTime
;
InFlightPacket
(
int
packetId
,
long
delayInMilliseconds
)
{
this
.
packetId
=
packetId
;
this
.
startTime
=
System
.
currentTimeMillis
()
+
delayInMilliseconds
;
}
@Override
public
long
getDelay
(
TimeUnit
unit
)
{
long
diff
=
startTime
-
System
.
currentTimeMillis
();
return
unit
.
convert
(
diff
,
TimeUnit
.
MILLISECONDS
);
}
@Override
public
int
compareTo
(
Delayed
o
)
{
if
((
this
.
startTime
-
((
InFlightPacket
)
o
).
startTime
)
==
0
)
{
return
0
;
}
if
((
this
.
startTime
-
((
InFlightPacket
)
o
).
startTime
)
>
0
)
{
return
1
;
}
else
{
return
-
1
;
}
}
}
public
MQTTSession
(
String
clientId
,
ClientRole
clientRole
,
Set
<
String
>
groups
,
boolean
isConnected
,
boolean
cleanSession
,
RemotingChannel
remotingChannel
,
long
lastUpdateTimestamp
,
DefaultMqttMessageProcessor
defaultMqttMessageProcessor
)
{
...
...
@@ -149,9 +116,10 @@ public class MQTTSession extends Client {
if
(
inflightSlots
.
get
()
>
0
)
{
inflightSlots
.
decrementAndGet
();
mqttHeader
.
setPacketId
(
getNextPacketId
());
inflightWindow
.
put
(
mqttHeader
.
getPacketId
(),
new
InFlightMessage
(
mqttHeader
.
getTopicName
(),
mqttHeader
.
getQosLevel
(),
messageExt
.
getBody
(),
brokerData
,
messageExt
.
getMsgId
(),
messageExt
.
getQueueOffset
()));
// inflightTimeouts.add(new InFlightPacket(mqttHeader.getPacketId(), FLIGHT_BEFORE_RESEND_MS));
put2processTable
(((
IOTClientManagerImpl
)
this
.
defaultMqttMessageProcessor
.
getIotClientManager
()).
getProcessTable
(),
brokerData
.
getBrokerName
(),
MqttUtil
.
getRootTopic
(
mqttHeader
.
getTopicName
()),
messageExt
);
inflightWindow
.
put
(
mqttHeader
.
getPacketId
(),
new
InFlightMessage
(
mqttHeader
.
getTopicName
(),
mqttHeader
.
getQosLevel
(),
messageExt
.
getBody
(),
brokerData
,
messageExt
.
getQueueOffset
()));
IOTClientManagerImpl
iotClientManager
=
(
IOTClientManagerImpl
)
this
.
defaultMqttMessageProcessor
.
getIotClientManager
();
iotClientManager
.
getInflightTimeouts
().
add
(
new
InFlightPacket
(
this
,
mqttHeader
.
getPacketId
(),
FLIGHT_BEFORE_RESEND_MS
));
put2processTable
(
iotClientManager
.
getProcessTable
(),
brokerData
.
getBrokerName
(),
MqttUtil
.
getRootTopic
(
mqttHeader
.
getTopicName
()),
messageExt
);
pushMessage2Client
(
mqttHeader
,
messageExt
.
getBody
());
}
}
...
...
@@ -168,11 +136,12 @@ public class MQTTSession extends Client {
}
}
inflightSlots
.
incrementAndGet
();
((
IOTClientManagerImpl
)
this
.
defaultMqttMessageProcessor
.
getIotClientManager
()).
getInflightTimeouts
().
remove
(
new
InFlightPacket
(
this
,
ackPacketId
,
0
));
releasePacketId
(
ackPacketId
);
return
remove
;
}
p
rivate
void
pushMessage2Client
(
MqttHeader
mqttHeader
,
byte
[]
body
)
{
p
ublic
void
pushMessage2Client
(
MqttHeader
mqttHeader
,
byte
[]
body
)
{
try
{
//set remaining length
int
remainingLength
=
mqttHeader
.
getTopicName
().
getBytes
().
length
+
body
.
length
;
...
...
@@ -259,10 +228,6 @@ public class MQTTSession extends Client {
return
inflightWindow
;
}
public
DelayQueue
<
InFlightPacket
>
getInflightTimeouts
()
{
return
inflightTimeouts
;
}
public
Hashtable
getInUsePacketIds
()
{
return
inUsePacketIds
;
}
...
...
mqtt/src/main/java/org/apache/rocketmq/mqtt/constant/MqttConstant.java
浏览文件 @
f0ad359c
...
...
@@ -27,6 +27,7 @@ public class MqttConstant {
public
static
final
String
SUBSCRIPTION_SEPARATOR
=
"/"
;
public
static
final
String
TOPIC_CLIENTID_SEPARATOR
=
"@"
;
public
static
final
long
DEFAULT_TIMEOUT_MILLS
=
3000L
;
public
static
final
int
FLIGHT_BEFORE_RESEND_MS
=
5_000
;
public
static
final
String
PROPERTY_MQTT_QOS
=
"PROPERTY_MQTT_QOS"
;
public
static
final
AttributeKey
<
Client
>
MQTT_CLIENT_ATTRIBUTE_KEY
=
AttributeKey
.
valueOf
(
"mqtt.client"
);
}
mqtt/src/main/java/org/apache/rocketmq/mqtt/service/impl/MqttScheduledServiceImpl.java
浏览文件 @
f0ad359c
...
...
@@ -16,6 +16,9 @@
*/
package
org.apache.rocketmq.mqtt.service.impl
;
import
io.netty.handler.codec.mqtt.MqttMessageType
;
import
java.util.ArrayList
;
import
java.util.Collection
;
import
java.util.Map
;
import
java.util.TreeMap
;
import
java.util.concurrent.ConcurrentHashMap
;
...
...
@@ -29,7 +32,12 @@ import org.apache.rocketmq.common.service.ScheduledService;
import
org.apache.rocketmq.logging.InternalLogger
;
import
org.apache.rocketmq.logging.InternalLoggerFactory
;
import
org.apache.rocketmq.mqtt.client.IOTClientManagerImpl
;
import
org.apache.rocketmq.mqtt.client.InFlightMessage
;
import
org.apache.rocketmq.mqtt.client.InFlightPacket
;
import
org.apache.rocketmq.mqtt.client.MQTTSession
;
import
org.apache.rocketmq.mqtt.constant.MqttConstant
;
import
org.apache.rocketmq.mqtt.processor.DefaultMqttMessageProcessor
;
import
org.apache.rocketmq.remoting.transport.mqtt.MqttHeader
;
public
class
MqttScheduledServiceImpl
implements
ScheduledService
{
private
static
final
InternalLogger
log
=
InternalLoggerFactory
.
getLogger
(
LoggerName
.
MQTT_LOGGER_NAME
);
...
...
@@ -67,6 +75,36 @@ public class MqttScheduledServiceImpl implements ScheduledService {
}
}
},
0
,
defaultMqttMessageProcessor
.
getMqttConfig
().
getPersistOffsetInterval
(),
TimeUnit
.
MILLISECONDS
);
this
.
mqttScheduledExecutorService
.
scheduleAtFixedRate
(
new
Runnable
()
{
@Override
public
void
run
()
{
IOTClientManagerImpl
iotClientManager
=
(
IOTClientManagerImpl
)
defaultMqttMessageProcessor
.
getIotClientManager
();
Collection
<
InFlightPacket
>
expired
=
new
ArrayList
<>();
iotClientManager
.
getInflightTimeouts
().
drainTo
(
expired
);
for
(
InFlightPacket
notAcked
:
expired
)
{
MQTTSession
client
=
notAcked
.
getClient
();
if
(!
client
.
isConnected
())
{
continue
;
}
if
(
notAcked
.
getResendTime
()
>
3
)
{
client
.
getRemotingChannel
().
close
();
continue
;
}
if
(
client
.
getInflightWindow
().
containsKey
(
notAcked
.
getPacketId
()))
{
InFlightMessage
inFlightMessage
=
client
.
getInflightWindow
().
get
(
notAcked
.
getPacketId
());
MqttHeader
mqttHeader
=
new
MqttHeader
();
mqttHeader
.
setTopicName
(
inFlightMessage
.
getTopic
());
mqttHeader
.
setQosLevel
(
inFlightMessage
.
getPushQos
());
mqttHeader
.
setRetain
(
false
);
mqttHeader
.
setDup
(
true
);
mqttHeader
.
setMessageType
(
MqttMessageType
.
PUBLISH
.
value
());
notAcked
.
setStartTime
(
System
.
currentTimeMillis
()
+
MqttConstant
.
FLIGHT_BEFORE_RESEND_MS
);
notAcked
.
setResendTime
(
notAcked
.
getResendTime
()
+
1
);
iotClientManager
.
getInflightTimeouts
().
add
(
notAcked
);
client
.
pushMessage2Client
(
mqttHeader
,
inFlightMessage
.
getBody
());
}
}
}
},
10000
,
defaultMqttMessageProcessor
.
getMqttConfig
().
getScanAckTimeoutInterval
(),
TimeUnit
.
MILLISECONDS
);
}
@Override
...
...
mqtt/src/main/java/org/apache/rocketmq/mqtt/task/MqttPushTask.java
浏览文件 @
f0ad359c
...
...
@@ -68,8 +68,8 @@ public class MqttPushTask implements Runnable {
private
BrokerData
brokerData
;
private
String
rootTopic
;
public
MqttPushTask
(
DefaultMqttMessageProcessor
processor
,
final
MqttHeader
mqttHeader
,
String
rootTopic
,
Client
client
,
BrokerData
brokerData
)
{
public
MqttPushTask
(
DefaultMqttMessageProcessor
processor
,
final
MqttHeader
mqttHeader
,
String
rootTopic
,
Client
client
,
BrokerData
brokerData
)
{
this
.
defaultMqttMessageProcessor
=
processor
;
this
.
mqttHeader
=
mqttHeader
;
this
.
rootTopic
=
rootTopic
;
...
...
mqtt/src/test/java/org/apache/rocketmq/mqtt/MqttPubackMessageHandlerTest.java
浏览文件 @
f0ad359c
...
...
@@ -84,7 +84,7 @@ public class MqttPubackMessageHandlerTest {
MQTTSession
mqttSession
=
Mockito
.
spy
(
new
MQTTSession
(
"client1"
,
ClientRole
.
IOTCLIENT
,
null
,
true
,
true
,
remotingChannel
,
System
.
currentTimeMillis
(),
defaultMqttMessageProcessor
));
Mockito
.
when
(
iotClientManager
.
getClient
(
anyString
(),
any
(
RemotingChannel
.
class
))).
thenReturn
(
mqttSession
);
InFlightMessage
inFlightMessage
=
Mockito
.
spy
(
new
InFlightMessage
(
"topicTest"
,
0
,
"Hello"
.
getBytes
(),
null
,
null
,
0
));
InFlightMessage
inFlightMessage
=
Mockito
.
spy
(
new
InFlightMessage
(
"topicTest"
,
0
,
"Hello"
.
getBytes
(),
null
,
0
));
doReturn
(
inFlightMessage
).
when
(
mqttSession
).
pubAckReceived
(
anyInt
());
RemotingCommand
remotingCommand
=
mqttPubackMessageHandler
.
handleMessage
(
mqttMessage
,
remotingChannel
);
assert
remotingCommand
==
null
;
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录