Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
小五666\n哈哈
Rocketmq
提交
4dbdbf0a
R
Rocketmq
项目概览
小五666\n哈哈
/
Rocketmq
与 Fork 源项目一致
Fork自
Apache RocketMQ / Rocketmq
通知
1
Star
0
Fork
0
代码
文件
提交
分支
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看板
未验证
提交
4dbdbf0a
编写于
9月 25, 2021
作者:
youlixishia
提交者:
GitHub
9月 25, 2021
浏览文件
操作
浏览文件
下载
差异文件
Merge pull request #3357 from Jason918/RIP-7
RIP-7 Multiple Directories Storage Support
上级
538b6b7f
14d50f98
变更
9
显示空白变更内容
内联
并排
Showing
9 changed file
with
492 addition
and
65 deletion
+492
-65
broker/src/main/java/org/apache/rocketmq/broker/processor/SendMessageProcessor.java
...pache/rocketmq/broker/processor/SendMessageProcessor.java
+8
-3
common/src/main/java/org/apache/rocketmq/common/UtilAll.java
common/src/main/java/org/apache/rocketmq/common/UtilAll.java
+5
-0
store/src/main/java/org/apache/rocketmq/store/CommitLog.java
store/src/main/java/org/apache/rocketmq/store/CommitLog.java
+24
-2
store/src/main/java/org/apache/rocketmq/store/DefaultMessageStore.java
...n/java/org/apache/rocketmq/store/DefaultMessageStore.java
+58
-13
store/src/main/java/org/apache/rocketmq/store/MappedFileQueue.java
.../main/java/org/apache/rocketmq/store/MappedFileQueue.java
+61
-47
store/src/main/java/org/apache/rocketmq/store/MultiPathMappedFileQueue.java
...a/org/apache/rocketmq/store/MultiPathMappedFileQueue.java
+127
-0
store/src/main/java/org/apache/rocketmq/store/config/MessageStoreConfig.java
.../org/apache/rocketmq/store/config/MessageStoreConfig.java
+13
-0
store/src/test/java/org/apache/rocketmq/store/DefaultMessageStoreCleanFilesTest.java
...che/rocketmq/store/DefaultMessageStoreCleanFilesTest.java
+42
-0
store/src/test/java/org/apache/rocketmq/store/MultiPathMappedFileQueueTest.java
...g/apache/rocketmq/store/MultiPathMappedFileQueueTest.java
+154
-0
未找到文件。
broker/src/main/java/org/apache/rocketmq/broker/processor/SendMessageProcessor.java
浏览文件 @
4dbdbf0a
...
@@ -56,6 +56,7 @@ import org.apache.rocketmq.remoting.netty.RemotingResponseCallback;
...
@@ -56,6 +56,7 @@ import org.apache.rocketmq.remoting.netty.RemotingResponseCallback;
import
org.apache.rocketmq.remoting.protocol.RemotingCommand
;
import
org.apache.rocketmq.remoting.protocol.RemotingCommand
;
import
org.apache.rocketmq.store.MessageExtBrokerInner
;
import
org.apache.rocketmq.store.MessageExtBrokerInner
;
import
org.apache.rocketmq.store.PutMessageResult
;
import
org.apache.rocketmq.store.PutMessageResult
;
import
org.apache.rocketmq.store.config.MessageStoreConfig
;
import
org.apache.rocketmq.store.config.StorePathConfigHelper
;
import
org.apache.rocketmq.store.config.StorePathConfigHelper
;
import
org.apache.rocketmq.store.stats.BrokerStatsManager
;
import
org.apache.rocketmq.store.stats.BrokerStatsManager
;
...
@@ -636,8 +637,12 @@ public class SendMessageProcessor extends AbstractSendMessageProcessor implement
...
@@ -636,8 +637,12 @@ public class SendMessageProcessor extends AbstractSendMessageProcessor implement
}
}
private
String
diskUtil
()
{
private
String
diskUtil
()
{
String
storePathPhysic
=
this
.
brokerController
.
getMessageStoreConfig
().
getStorePathCommitLog
();
double
physicRatio
=
100
;
double
physicRatio
=
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
storePathPhysic
);
String
storePath
=
this
.
brokerController
.
getMessageStoreConfig
().
getStorePathCommitLog
();
String
[]
paths
=
storePath
.
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
for
(
String
storePathPhysic
:
paths
)
{
physicRatio
=
Math
.
min
(
physicRatio
,
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
storePathPhysic
));
}
String
storePathLogis
=
String
storePathLogis
=
StorePathConfigHelper
.
getStorePathConsumeQueue
(
this
.
brokerController
.
getMessageStoreConfig
().
getStorePathRootDir
());
StorePathConfigHelper
.
getStorePathConsumeQueue
(
this
.
brokerController
.
getMessageStoreConfig
().
getStorePathRootDir
());
...
...
common/src/main/java/org/apache/rocketmq/common/UtilAll.java
浏览文件 @
4dbdbf0a
...
@@ -198,6 +198,11 @@ public class UtilAll {
...
@@ -198,6 +198,11 @@ public class UtilAll {
cal
.
get
(
Calendar
.
SECOND
));
cal
.
get
(
Calendar
.
SECOND
));
}
}
public
static
boolean
isPathExists
(
final
String
path
)
{
File
file
=
new
File
(
path
);
return
file
.
exists
();
}
public
static
double
getDiskPartitionSpaceUsedPercent
(
final
String
path
)
{
public
static
double
getDiskPartitionSpaceUsedPercent
(
final
String
path
)
{
if
(
null
==
path
||
path
.
isEmpty
())
{
if
(
null
==
path
||
path
.
isEmpty
())
{
log
.
error
(
"Error when measuring disk space usage, path is null or empty, path : {}"
,
path
);
log
.
error
(
"Error when measuring disk space usage, path is null or empty, path : {}"
,
path
);
...
...
store/src/main/java/org/apache/rocketmq/store/CommitLog.java
浏览文件 @
4dbdbf0a
...
@@ -22,10 +22,12 @@ import java.net.InetAddress;
...
@@ -22,10 +22,12 @@ import java.net.InetAddress;
import
java.net.InetSocketAddress
;
import
java.net.InetSocketAddress
;
import
java.net.SocketAddress
;
import
java.net.SocketAddress
;
import
java.nio.ByteBuffer
;
import
java.nio.ByteBuffer
;
import
java.util.Collections
;
import
java.util.HashMap
;
import
java.util.HashMap
;
import
java.util.LinkedList
;
import
java.util.LinkedList
;
import
java.util.List
;
import
java.util.List
;
import
java.util.Map
;
import
java.util.Map
;
import
java.util.Set
;
import
java.util.concurrent.CompletableFuture
;
import
java.util.concurrent.CompletableFuture
;
import
java.util.function.Supplier
;
import
java.util.function.Supplier
;
...
@@ -43,6 +45,7 @@ import org.apache.rocketmq.logging.InternalLogger;
...
@@ -43,6 +45,7 @@ import org.apache.rocketmq.logging.InternalLogger;
import
org.apache.rocketmq.logging.InternalLoggerFactory
;
import
org.apache.rocketmq.logging.InternalLoggerFactory
;
import
org.apache.rocketmq.store.config.BrokerRole
;
import
org.apache.rocketmq.store.config.BrokerRole
;
import
org.apache.rocketmq.store.config.FlushDiskType
;
import
org.apache.rocketmq.store.config.FlushDiskType
;
import
org.apache.rocketmq.store.config.MessageStoreConfig
;
import
org.apache.rocketmq.store.ha.HAService
;
import
org.apache.rocketmq.store.ha.HAService
;
import
org.apache.rocketmq.store.schedule.ScheduleMessageService
;
import
org.apache.rocketmq.store.schedule.ScheduleMessageService
;
...
@@ -71,9 +74,20 @@ public class CommitLog {
...
@@ -71,9 +74,20 @@ public class CommitLog {
protected
final
PutMessageLock
putMessageLock
;
protected
final
PutMessageLock
putMessageLock
;
private
volatile
Set
<
String
>
fullStorePaths
=
Collections
.
emptySet
();
public
CommitLog
(
final
DefaultMessageStore
defaultMessageStore
)
{
public
CommitLog
(
final
DefaultMessageStore
defaultMessageStore
)
{
this
.
mappedFileQueue
=
new
MappedFileQueue
(
defaultMessageStore
.
getMessageStoreConfig
().
getStorePathCommitLog
(),
String
storePath
=
defaultMessageStore
.
getMessageStoreConfig
().
getStorePathCommitLog
();
defaultMessageStore
.
getMessageStoreConfig
().
getMappedFileSizeCommitLog
(),
defaultMessageStore
.
getAllocateMappedFileService
());
if
(
storePath
.
contains
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
))
{
this
.
mappedFileQueue
=
new
MultiPathMappedFileQueue
(
defaultMessageStore
.
getMessageStoreConfig
(),
defaultMessageStore
.
getMessageStoreConfig
().
getMappedFileSizeCommitLog
(),
defaultMessageStore
.
getAllocateMappedFileService
(),
this
::
getFullStorePaths
);
}
else
{
this
.
mappedFileQueue
=
new
MappedFileQueue
(
storePath
,
defaultMessageStore
.
getMessageStoreConfig
().
getMappedFileSizeCommitLog
(),
defaultMessageStore
.
getAllocateMappedFileService
());
}
this
.
defaultMessageStore
=
defaultMessageStore
;
this
.
defaultMessageStore
=
defaultMessageStore
;
if
(
FlushDiskType
.
SYNC_FLUSH
==
defaultMessageStore
.
getMessageStoreConfig
().
getFlushDiskType
())
{
if
(
FlushDiskType
.
SYNC_FLUSH
==
defaultMessageStore
.
getMessageStoreConfig
().
getFlushDiskType
())
{
...
@@ -95,6 +109,14 @@ public class CommitLog {
...
@@ -95,6 +109,14 @@ public class CommitLog {
}
}
public
void
setFullStorePaths
(
Set
<
String
>
fullStorePaths
)
{
this
.
fullStorePaths
=
fullStorePaths
;
}
public
Set
<
String
>
getFullStorePaths
()
{
return
fullStorePaths
;
}
public
boolean
load
()
{
public
boolean
load
()
{
boolean
result
=
this
.
mappedFileQueue
.
load
();
boolean
result
=
this
.
mappedFileQueue
.
load
();
log
.
info
(
"load commit log "
+
(
result
?
"OK"
:
"Failed"
));
log
.
info
(
"load commit log "
+
(
result
?
"OK"
:
"Failed"
));
...
...
store/src/main/java/org/apache/rocketmq/store/DefaultMessageStore.java
浏览文件 @
4dbdbf0a
...
@@ -26,6 +26,7 @@ import java.nio.ByteBuffer;
...
@@ -26,6 +26,7 @@ import java.nio.ByteBuffer;
import
java.nio.channels.FileLock
;
import
java.nio.channels.FileLock
;
import
java.util.Collections
;
import
java.util.Collections
;
import
java.util.HashMap
;
import
java.util.HashMap
;
import
java.util.HashSet
;
import
java.util.Iterator
;
import
java.util.Iterator
;
import
java.util.LinkedList
;
import
java.util.LinkedList
;
import
java.util.Map
;
import
java.util.Map
;
...
@@ -784,15 +785,22 @@ public class DefaultMessageStore implements MessageStore {
...
@@ -784,15 +785,22 @@ public class DefaultMessageStore implements MessageStore {
HashMap
<
String
,
String
>
result
=
this
.
storeStatsService
.
getRuntimeInfo
();
HashMap
<
String
,
String
>
result
=
this
.
storeStatsService
.
getRuntimeInfo
();
{
{
double
physicRatio
=
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
getStorePathPhysic
());
double
minPhysicsUsedRatio
=
Double
.
MAX_VALUE
;
result
.
put
(
RunningStats
.
commitLogDiskRatio
.
name
(),
String
.
valueOf
(
physicRatio
));
String
commitLogStorePath
=
getStorePathPhysic
();
String
[]
paths
=
commitLogStorePath
.
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
for
(
String
clPath
:
paths
)
{
double
physicRatio
=
UtilAll
.
isPathExists
(
clPath
)
?
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
clPath
)
:
-
1
;
result
.
put
(
RunningStats
.
commitLogDiskRatio
.
name
()
+
"_"
+
clPath
,
String
.
valueOf
(
physicRatio
));
minPhysicsUsedRatio
=
Math
.
min
(
minPhysicsUsedRatio
,
physicRatio
);
}
result
.
put
(
RunningStats
.
commitLogDiskRatio
.
name
(),
String
.
valueOf
(
minPhysicsUsedRatio
));
}
}
{
{
String
storePathLogics
=
StorePathConfigHelper
.
getStorePathConsumeQueue
(
this
.
messageStoreConfig
.
getStorePathRootDir
());
String
storePathLogics
=
StorePathConfigHelper
.
getStorePathConsumeQueue
(
this
.
messageStoreConfig
.
getStorePathRootDir
());
double
logicsRatio
=
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
storePathLogics
);
double
logicsRatio
=
UtilAll
.
isPathExists
(
storePathLogics
)
?
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
storePathLogics
)
:
-
1
;
result
.
put
(
RunningStats
.
consumeQueueDiskRatio
.
name
(),
String
.
valueOf
(
logicsRatio
));
result
.
put
(
RunningStats
.
consumeQueueDiskRatio
.
name
(),
String
.
valueOf
(
logicsRatio
));
}
}
...
@@ -1651,25 +1659,43 @@ public class DefaultMessageStore implements MessageStore {
...
@@ -1651,25 +1659,43 @@ public class DefaultMessageStore implements MessageStore {
cleanImmediately
=
false
;
cleanImmediately
=
false
;
{
{
double
physicRatio
=
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
getStorePathPhysic
());
String
commitLogStorePath
=
DefaultMessageStore
.
this
.
getMessageStoreConfig
().
getStorePathCommitLog
();
if
(
physicRatio
>
diskSpaceWarningLevelRatio
)
{
String
[]
storePaths
=
commitLogStorePath
.
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
Set
<
String
>
fullStorePath
=
new
HashSet
<>();
double
minPhysicRatio
=
100
;
String
minStorePath
=
null
;
for
(
String
storePathPhysic
:
storePaths
)
{
double
physicRatio
=
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
storePathPhysic
);
if
(
minPhysicRatio
>
physicRatio
)
{
minPhysicRatio
=
physicRatio
;
minStorePath
=
storePathPhysic
;
}
if
(
physicRatio
>
diskSpaceCleanForciblyRatio
)
{
fullStorePath
.
add
(
storePathPhysic
);
}
}
DefaultMessageStore
.
this
.
commitLog
.
setFullStorePaths
(
fullStorePath
);
if
(
minPhysicRatio
>
diskSpaceWarningLevelRatio
)
{
boolean
diskok
=
DefaultMessageStore
.
this
.
runningFlags
.
getAndMakeDiskFull
();
boolean
diskok
=
DefaultMessageStore
.
this
.
runningFlags
.
getAndMakeDiskFull
();
if
(
diskok
)
{
if
(
diskok
)
{
DefaultMessageStore
.
log
.
error
(
"physic disk maybe full soon "
+
physicRatio
+
", so mark disk full"
);
DefaultMessageStore
.
log
.
error
(
"physic disk maybe full soon "
+
minPhysicRatio
+
", so mark disk full, storePathPhysic="
+
minStorePath
);
}
}
cleanImmediately
=
true
;
cleanImmediately
=
true
;
}
else
if
(
p
hysicRatio
>
diskSpaceCleanForciblyRatio
)
{
}
else
if
(
minP
hysicRatio
>
diskSpaceCleanForciblyRatio
)
{
cleanImmediately
=
true
;
cleanImmediately
=
true
;
}
else
{
}
else
{
boolean
diskok
=
DefaultMessageStore
.
this
.
runningFlags
.
getAndMakeDiskOK
();
boolean
diskok
=
DefaultMessageStore
.
this
.
runningFlags
.
getAndMakeDiskOK
();
if
(!
diskok
)
{
if
(!
diskok
)
{
DefaultMessageStore
.
log
.
info
(
"physic disk space OK "
+
physicRatio
+
", so mark disk ok"
);
DefaultMessageStore
.
log
.
info
(
"physic disk space OK "
+
minPhysicRatio
+
", so mark disk ok, storePathPhysic="
+
minStorePath
);
}
}
}
}
if
(
physicRatio
<
0
||
physicRatio
>
ratio
)
{
if
(
minPhysicRatio
<
0
||
minPhysicRatio
>
ratio
)
{
DefaultMessageStore
.
log
.
info
(
"physic disk maybe full soon, so reclaim space, "
+
physicRatio
);
DefaultMessageStore
.
log
.
info
(
"physic disk maybe full soon, so reclaim space, "
+
minPhysicRatio
+
", storePathPhysic="
+
minStorePath
);
return
true
;
return
true
;
}
}
}
}
...
@@ -1710,8 +1736,27 @@ public class DefaultMessageStore implements MessageStore {
...
@@ -1710,8 +1736,27 @@ public class DefaultMessageStore implements MessageStore {
public
void
setManualDeleteFileSeveralTimes
(
int
manualDeleteFileSeveralTimes
)
{
public
void
setManualDeleteFileSeveralTimes
(
int
manualDeleteFileSeveralTimes
)
{
this
.
manualDeleteFileSeveralTimes
=
manualDeleteFileSeveralTimes
;
this
.
manualDeleteFileSeveralTimes
=
manualDeleteFileSeveralTimes
;
}
}
public
double
calcStorePathPhysicRatio
()
{
Set
<
String
>
fullStorePath
=
new
HashSet
<>();
String
storePath
=
getStorePathPhysic
();
String
[]
paths
=
storePath
.
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
double
minPhysicRatio
=
100
;
for
(
String
path
:
paths
)
{
double
physicRatio
=
UtilAll
.
isPathExists
(
path
)
?
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
path
)
:
-
1
;
minPhysicRatio
=
Math
.
min
(
minPhysicRatio
,
physicRatio
);
if
(
physicRatio
>
diskSpaceCleanForciblyRatio
)
{
fullStorePath
.
add
(
path
);
}
}
DefaultMessageStore
.
this
.
commitLog
.
setFullStorePaths
(
fullStorePath
);
return
minPhysicRatio
;
}
public
boolean
isSpaceFull
()
{
public
boolean
isSpaceFull
()
{
double
physicRatio
=
UtilAll
.
getDiskPartitionSpaceUsedPercent
(
getStorePathPhysic
()
);
double
physicRatio
=
calcStorePathPhysicRatio
(
);
double
ratio
=
DefaultMessageStore
.
this
.
getMessageStoreConfig
().
getDiskMaxUsedSpaceRatio
()
/
100.0
;
double
ratio
=
DefaultMessageStore
.
this
.
getMessageStoreConfig
().
getDiskMaxUsedSpaceRatio
()
/
100.0
;
if
(
physicRatio
>
ratio
)
{
if
(
physicRatio
>
ratio
)
{
DefaultMessageStore
.
log
.
info
(
"physic disk of commitLog used: "
+
physicRatio
);
DefaultMessageStore
.
log
.
info
(
"physic disk of commitLog used: "
+
physicRatio
);
...
...
store/src/main/java/org/apache/rocketmq/store/MappedFileQueue.java
浏览文件 @
4dbdbf0a
...
@@ -20,6 +20,7 @@ import java.io.File;
...
@@ -20,6 +20,7 @@ import java.io.File;
import
java.io.IOException
;
import
java.io.IOException
;
import
java.util.ArrayList
;
import
java.util.ArrayList
;
import
java.util.Arrays
;
import
java.util.Arrays
;
import
java.util.Comparator
;
import
java.util.Iterator
;
import
java.util.Iterator
;
import
java.util.List
;
import
java.util.List
;
import
java.util.ListIterator
;
import
java.util.ListIterator
;
...
@@ -37,13 +38,13 @@ public class MappedFileQueue {
...
@@ -37,13 +38,13 @@ public class MappedFileQueue {
private
final
String
storePath
;
private
final
String
storePath
;
pr
ivate
final
int
mappedFileSize
;
pr
otected
final
int
mappedFileSize
;
pr
ivate
final
CopyOnWriteArrayList
<
MappedFile
>
mappedFiles
=
new
CopyOnWriteArrayList
<
MappedFile
>();
pr
otected
final
CopyOnWriteArrayList
<
MappedFile
>
mappedFiles
=
new
CopyOnWriteArrayList
<
MappedFile
>();
private
final
AllocateMappedFileService
allocateMappedFileService
;
private
final
AllocateMappedFileService
allocateMappedFileService
;
pr
ivate
long
flushedWhere
=
0
;
pr
otected
long
flushedWhere
=
0
;
private
long
committedWhere
=
0
;
private
long
committedWhere
=
0
;
private
volatile
long
storeTimestamp
=
0
;
private
volatile
long
storeTimestamp
=
0
;
...
@@ -144,18 +145,25 @@ public class MappedFileQueue {
...
@@ -144,18 +145,25 @@ public class MappedFileQueue {
}
}
}
}
public
boolean
load
()
{
public
boolean
load
()
{
File
dir
=
new
File
(
this
.
storePath
);
File
dir
=
new
File
(
this
.
storePath
);
File
[]
files
=
dir
.
listFiles
();
File
[]
ls
=
dir
.
listFiles
();
if
(
files
!=
null
)
{
if
(
ls
!=
null
)
{
return
doLoad
(
Arrays
.
asList
(
ls
));
}
return
true
;
}
public
boolean
doLoad
(
List
<
File
>
files
)
{
// ascending order
// ascending order
Arrays
.
sort
(
files
);
files
.
sort
(
Comparator
.
comparing
(
File:
:
getName
));
for
(
File
file
:
files
)
{
for
(
File
file
:
files
)
{
if
(
file
.
length
()
!=
this
.
mappedFileSize
)
{
if
(
file
.
length
()
!=
this
.
mappedFileSize
)
{
log
.
warn
(
file
+
"\t"
+
file
.
length
()
log
.
warn
(
file
+
"\t"
+
file
.
length
()
+
" length not matched message store config value,
please check it manually
"
);
+
" length not matched message store config value,
ignore it
"
);
return
fals
e
;
return
tru
e
;
}
}
try
{
try
{
...
@@ -171,8 +179,6 @@ public class MappedFileQueue {
...
@@ -171,8 +179,6 @@ public class MappedFileQueue {
return
false
;
return
false
;
}
}
}
}
}
return
true
;
return
true
;
}
}
...
@@ -204,9 +210,20 @@ public class MappedFileQueue {
...
@@ -204,9 +210,20 @@ public class MappedFileQueue {
}
}
if
(
createOffset
!=
-
1
&&
needCreate
)
{
if
(
createOffset
!=
-
1
&&
needCreate
)
{
return
tryCreateMappedFile
(
createOffset
);
}
return
mappedFileLast
;
}
protected
MappedFile
tryCreateMappedFile
(
long
createOffset
)
{
String
nextFilePath
=
this
.
storePath
+
File
.
separator
+
UtilAll
.
offset2FileName
(
createOffset
);
String
nextFilePath
=
this
.
storePath
+
File
.
separator
+
UtilAll
.
offset2FileName
(
createOffset
);
String
nextNextFilePath
=
this
.
storePath
+
File
.
separator
String
nextNextFilePath
=
this
.
storePath
+
File
.
separator
+
UtilAll
.
offset2FileName
(
createOffset
+
UtilAll
.
offset2FileName
(
createOffset
+
this
.
mappedFileSize
);
+
this
.
mappedFileSize
);
return
doCreateMappedFile
(
nextFilePath
,
nextNextFilePath
);
}
protected
MappedFile
doCreateMappedFile
(
String
nextFilePath
,
String
nextNextFilePath
)
{
MappedFile
mappedFile
=
null
;
MappedFile
mappedFile
=
null
;
if
(
this
.
allocateMappedFileService
!=
null
)
{
if
(
this
.
allocateMappedFileService
!=
null
)
{
...
@@ -230,9 +247,6 @@ public class MappedFileQueue {
...
@@ -230,9 +247,6 @@ public class MappedFileQueue {
return
mappedFile
;
return
mappedFile
;
}
}
return
mappedFileLast
;
}
public
MappedFile
getLastMappedFile
(
final
long
startOffset
)
{
public
MappedFile
getLastMappedFile
(
final
long
startOffset
)
{
return
getLastMappedFile
(
startOffset
,
true
);
return
getLastMappedFile
(
startOffset
,
true
);
}
}
...
...
store/src/main/java/org/apache/rocketmq/store/MultiPathMappedFileQueue.java
0 → 100644
浏览文件 @
4dbdbf0a
/*
* 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.store
;
import
java.util.Arrays
;
import
java.util.HashSet
;
import
java.util.Set
;
import
java.util.function.Supplier
;
import
org.apache.commons.lang3.StringUtils
;
import
org.apache.rocketmq.common.UtilAll
;
import
org.apache.rocketmq.store.config.MessageStoreConfig
;
import
java.io.File
;
import
java.util.ArrayList
;
import
java.util.Collections
;
import
java.util.List
;
public
class
MultiPathMappedFileQueue
extends
MappedFileQueue
{
private
final
MessageStoreConfig
config
;
private
final
Supplier
<
Set
<
String
>>
fullStorePathsSupplier
;
public
MultiPathMappedFileQueue
(
MessageStoreConfig
messageStoreConfig
,
int
mappedFileSize
,
AllocateMappedFileService
allocateMappedFileService
,
Supplier
<
Set
<
String
>>
fullStorePathsSupplier
)
{
super
(
messageStoreConfig
.
getStorePathCommitLog
(),
mappedFileSize
,
allocateMappedFileService
);
this
.
config
=
messageStoreConfig
;
this
.
fullStorePathsSupplier
=
fullStorePathsSupplier
;
}
private
Set
<
String
>
getPaths
()
{
String
[]
paths
=
config
.
getStorePathCommitLog
().
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
return
new
HashSet
<>(
Arrays
.
asList
(
paths
));
}
private
Set
<
String
>
getReadonlyPaths
()
{
String
pathStr
=
config
.
getReadOnlyCommitLogStorePaths
();
if
(
StringUtils
.
isBlank
(
pathStr
))
{
return
Collections
.
emptySet
();
}
String
[]
paths
=
pathStr
.
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
return
new
HashSet
<>(
Arrays
.
asList
(
paths
));
}
@Override
public
boolean
load
()
{
Set
<
String
>
storePathSet
=
getPaths
();
storePathSet
.
addAll
(
getReadonlyPaths
());
List
<
File
>
files
=
new
ArrayList
<>();
for
(
String
path
:
storePathSet
)
{
File
dir
=
new
File
(
path
);
File
[]
ls
=
dir
.
listFiles
();
if
(
ls
!=
null
)
{
Collections
.
addAll
(
files
,
ls
);
}
}
return
doLoad
(
files
);
}
@Override
protected
MappedFile
tryCreateMappedFile
(
long
createOffset
)
{
long
fileIdx
=
createOffset
/
this
.
mappedFileSize
;
Set
<
String
>
storePath
=
getPaths
();
Set
<
String
>
readonlyPathSet
=
getReadonlyPaths
();
Set
<
String
>
fullStorePaths
=
fullStorePathsSupplier
==
null
?
Collections
.
emptySet
()
:
fullStorePathsSupplier
.
get
();
HashSet
<
String
>
availableStorePath
=
new
HashSet
<>(
storePath
);
//do not create file in readonly store path.
availableStorePath
.
removeAll
(
readonlyPathSet
);
//do not create file is space is nearly full.
availableStorePath
.
removeAll
(
fullStorePaths
);
//if no store path left, fall back to writable store path.
if
(
availableStorePath
.
isEmpty
())
{
availableStorePath
=
new
HashSet
<>(
storePath
);
availableStorePath
.
removeAll
(
readonlyPathSet
);
}
String
[]
paths
=
availableStorePath
.
toArray
(
new
String
[]{});
Arrays
.
sort
(
paths
);
String
nextFilePath
=
paths
[(
int
)
(
fileIdx
%
paths
.
length
)]
+
File
.
separator
+
UtilAll
.
offset2FileName
(
createOffset
);
String
nextNextFilePath
=
paths
[(
int
)
((
fileIdx
+
1
)
%
paths
.
length
)]
+
File
.
separator
+
UtilAll
.
offset2FileName
(
createOffset
+
this
.
mappedFileSize
);
return
doCreateMappedFile
(
nextFilePath
,
nextNextFilePath
);
}
@Override
public
void
destroy
()
{
for
(
MappedFile
mf
:
this
.
mappedFiles
)
{
mf
.
destroy
(
1000
*
3
);
}
this
.
mappedFiles
.
clear
();
this
.
flushedWhere
=
0
;
Set
<
String
>
storePathSet
=
getPaths
();
storePathSet
.
addAll
(
getReadonlyPaths
());
for
(
String
path
:
storePathSet
)
{
File
file
=
new
File
(
path
);
if
(
file
.
isDirectory
())
{
file
.
delete
();
}
}
}
}
store/src/main/java/org/apache/rocketmq/store/config/MessageStoreConfig.java
浏览文件 @
4dbdbf0a
...
@@ -17,10 +17,14 @@
...
@@ -17,10 +17,14 @@
package
org.apache.rocketmq.store.config
;
package
org.apache.rocketmq.store.config
;
import
java.io.File
;
import
java.io.File
;
import
org.apache.rocketmq.common.annotation.ImportantField
;
import
org.apache.rocketmq.common.annotation.ImportantField
;
import
org.apache.rocketmq.store.ConsumeQueue
;
import
org.apache.rocketmq.store.ConsumeQueue
;
public
class
MessageStoreConfig
{
public
class
MessageStoreConfig
{
public
static
final
String
MULTI_PATH_SPLITTER
=
System
.
getProperty
(
"rocketmq.broker.multiPathSplitter"
,
","
);
//The root directory in which the log data is kept
//The root directory in which the log data is kept
@ImportantField
@ImportantField
private
String
storePathRootDir
=
System
.
getProperty
(
"user.home"
)
+
File
.
separator
+
"store"
;
private
String
storePathRootDir
=
System
.
getProperty
(
"user.home"
)
+
File
.
separator
+
"store"
;
...
@@ -30,6 +34,8 @@ public class MessageStoreConfig {
...
@@ -30,6 +34,8 @@ public class MessageStoreConfig {
private
String
storePathCommitLog
=
System
.
getProperty
(
"user.home"
)
+
File
.
separator
+
"store"
private
String
storePathCommitLog
=
System
.
getProperty
(
"user.home"
)
+
File
.
separator
+
"store"
+
File
.
separator
+
"commitlog"
;
+
File
.
separator
+
"commitlog"
;
private
String
readOnlyCommitLogStorePaths
=
null
;
// CommitLog file size,default is 1G
// CommitLog file size,default is 1G
private
int
mappedFileSizeCommitLog
=
1024
*
1024
*
1024
;
private
int
mappedFileSizeCommitLog
=
1024
*
1024
*
1024
;
// ConsumeQueue file size,default is 30W
// ConsumeQueue file size,default is 30W
...
@@ -676,6 +682,13 @@ public class MessageStoreConfig {
...
@@ -676,6 +682,13 @@ public class MessageStoreConfig {
this
.
commitCommitLogThoroughInterval
=
commitCommitLogThoroughInterval
;
this
.
commitCommitLogThoroughInterval
=
commitCommitLogThoroughInterval
;
}
}
public
String
getReadOnlyCommitLogStorePaths
()
{
return
readOnlyCommitLogStorePaths
;
}
public
void
setReadOnlyCommitLogStorePaths
(
String
readOnlyCommitLogStorePaths
)
{
this
.
readOnlyCommitLogStorePaths
=
readOnlyCommitLogStorePaths
;
}
public
String
getdLegerGroup
()
{
public
String
getdLegerGroup
()
{
return
dLegerGroup
;
return
dLegerGroup
;
}
}
...
...
store/src/test/java/org/apache/rocketmq/store/DefaultMessageStoreCleanFilesTest.java
浏览文件 @
4dbdbf0a
...
@@ -94,6 +94,41 @@ public class DefaultMessageStoreCleanFilesTest {
...
@@ -94,6 +94,41 @@ public class DefaultMessageStoreCleanFilesTest {
}
}
@Test
public
void
testIsSpaceFullMultiCommitLogStorePath
()
throws
Exception
{
String
deleteWhen
=
"04"
;
// the min value of diskMaxUsedSpaceRatio.
int
diskMaxUsedSpaceRatio
=
1
;
// used to set disk-full flag
double
diskSpaceCleanForciblyRatio
=
0.01
D
;
MessageStoreConfig
config
=
genMessageStoreConfig
(
deleteWhen
,
diskMaxUsedSpaceRatio
);
String
storePath
=
config
.
getStorePathCommitLog
();
StringBuilder
storePathBuilder
=
new
StringBuilder
();
for
(
int
i
=
0
;
i
<
3
;
i
++)
{
storePathBuilder
.
append
(
storePath
).
append
(
i
).
append
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
}
config
.
setStorePathCommitLog
(
storePathBuilder
.
toString
());
String
[]
paths
=
config
.
getStorePathCommitLog
().
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
assertEquals
(
3
,
paths
.
length
);
initMessageStore
(
config
,
diskSpaceCleanForciblyRatio
);
// build and put 55 messages, exactly one message per CommitLog file.
buildAndPutMessagesToMessageStore
(
msgCount
);
MappedFileQueue
commitLogQueue
=
getMappedFileQueueCommitLog
();
assertEquals
(
fileCountCommitLog
,
commitLogQueue
.
getMappedFiles
().
size
());
int
fileCountConsumeQueue
=
getFileCountConsumeQueue
();
MappedFileQueue
consumeQueue
=
getMappedFileQueueConsumeQueue
();
assertEquals
(
fileCountConsumeQueue
,
consumeQueue
.
getMappedFiles
().
size
());
cleanCommitLogService
.
isSpaceFull
();
assertEquals
(
1
<<
4
,
messageStore
.
getRunningFlags
().
getFlagBits
()
&
(
1
<<
4
));
messageStore
.
shutdown
();
messageStore
.
destroy
();
}
@Test
@Test
public
void
testIsSpaceFullFunctionFull2Empty
()
throws
Exception
{
public
void
testIsSpaceFullFunctionFull2Empty
()
throws
Exception
{
String
deleteWhen
=
"04"
;
String
deleteWhen
=
"04"
;
...
@@ -421,6 +456,10 @@ public class DefaultMessageStoreCleanFilesTest {
...
@@ -421,6 +456,10 @@ public class DefaultMessageStoreCleanFilesTest {
}
}
private
void
initMessageStore
(
String
deleteWhen
,
int
diskMaxUsedSpaceRatio
,
double
diskSpaceCleanForciblyRatio
)
throws
Exception
{
private
void
initMessageStore
(
String
deleteWhen
,
int
diskMaxUsedSpaceRatio
,
double
diskSpaceCleanForciblyRatio
)
throws
Exception
{
initMessageStore
(
genMessageStoreConfig
(
deleteWhen
,
diskMaxUsedSpaceRatio
),
diskSpaceCleanForciblyRatio
);
}
private
MessageStoreConfig
genMessageStoreConfig
(
String
deleteWhen
,
int
diskMaxUsedSpaceRatio
)
{
MessageStoreConfig
messageStoreConfig
=
new
MessageStoreConfigForTest
();
MessageStoreConfig
messageStoreConfig
=
new
MessageStoreConfigForTest
();
messageStoreConfig
.
setMappedFileSizeCommitLog
(
mappedFileSize
);
messageStoreConfig
.
setMappedFileSizeCommitLog
(
mappedFileSize
);
messageStoreConfig
.
setMappedFileSizeConsumeQueue
(
mappedFileSize
);
messageStoreConfig
.
setMappedFileSizeConsumeQueue
(
mappedFileSize
);
...
@@ -442,7 +481,10 @@ public class DefaultMessageStoreCleanFilesTest {
...
@@ -442,7 +481,10 @@ public class DefaultMessageStoreCleanFilesTest {
String
storePathCommitLog
=
storePathRootDir
+
File
.
separator
+
"commitlog"
;
String
storePathCommitLog
=
storePathRootDir
+
File
.
separator
+
"commitlog"
;
messageStoreConfig
.
setStorePathRootDir
(
storePathRootDir
);
messageStoreConfig
.
setStorePathRootDir
(
storePathRootDir
);
messageStoreConfig
.
setStorePathCommitLog
(
storePathCommitLog
);
messageStoreConfig
.
setStorePathCommitLog
(
storePathCommitLog
);
return
messageStoreConfig
;
}
private
void
initMessageStore
(
MessageStoreConfig
messageStoreConfig
,
double
diskSpaceCleanForciblyRatio
)
throws
Exception
{
messageStore
=
new
DefaultMessageStore
(
messageStoreConfig
,
messageStore
=
new
DefaultMessageStore
(
messageStoreConfig
,
new
BrokerStatsManager
(
"test"
),
new
MyMessageArrivingListener
(),
new
BrokerConfig
());
new
BrokerStatsManager
(
"test"
),
new
MyMessageArrivingListener
(),
new
BrokerConfig
());
...
...
store/src/test/java/org/apache/rocketmq/store/MultiPathMappedFileQueueTest.java
0 → 100644
浏览文件 @
4dbdbf0a
/*
* 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.store
;
import
static
org
.
assertj
.
core
.
api
.
Assertions
.
assertThat
;
import
java.util.HashSet
;
import
java.util.Set
;
import
org.apache.rocketmq.common.UtilAll
;
import
org.apache.rocketmq.store.config.MessageStoreConfig
;
import
org.junit.Test
;
public
class
MultiPathMappedFileQueueTest
{
@Test
public
void
testGetLastMappedFile
()
{
final
byte
[]
fixedMsg
=
new
byte
[
1024
];
MessageStoreConfig
config
=
new
MessageStoreConfig
();
config
.
setStorePathCommitLog
(
"target/unit_test_store/a/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/b/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/c/"
);
MappedFileQueue
mappedFileQueue
=
new
MultiPathMappedFileQueue
(
config
,
1024
,
null
,
null
);
String
[]
storePaths
=
config
.
getStorePathCommitLog
().
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
for
(
int
i
=
0
;
i
<
1024
;
i
++)
{
MappedFile
mappedFile
=
mappedFileQueue
.
getLastMappedFile
(
fixedMsg
.
length
*
i
);
assertThat
(
mappedFile
).
isNotNull
();
assertThat
(
mappedFile
.
appendMessage
(
fixedMsg
)).
isTrue
();
int
idx
=
i
%
storePaths
.
length
;
assertThat
(
mappedFile
.
getFileName
().
startsWith
(
storePaths
[
idx
])).
isTrue
();
}
mappedFileQueue
.
shutdown
(
1000
);
mappedFileQueue
.
destroy
();
}
@Test
public
void
testLoadReadOnlyMappedFiles
()
{
{
//create old mapped files
final
byte
[]
fixedMsg
=
new
byte
[
1024
];
MessageStoreConfig
config
=
new
MessageStoreConfig
();
config
.
setStorePathCommitLog
(
"target/unit_test_store/a/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/b/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/c/"
);
MappedFileQueue
mappedFileQueue
=
new
MultiPathMappedFileQueue
(
config
,
1024
,
null
,
null
);
String
[]
storePaths
=
config
.
getStorePathCommitLog
().
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
for
(
int
i
=
0
;
i
<
1024
;
i
++)
{
MappedFile
mappedFile
=
mappedFileQueue
.
getLastMappedFile
(
fixedMsg
.
length
*
i
);
assertThat
(
mappedFile
).
isNotNull
();
assertThat
(
mappedFile
.
appendMessage
(
fixedMsg
)).
isTrue
();
int
idx
=
i
%
storePaths
.
length
;
assertThat
(
mappedFile
.
getFileName
().
startsWith
(
storePaths
[
idx
])).
isTrue
();
}
mappedFileQueue
.
shutdown
(
1000
);
}
// test load and readonly
MessageStoreConfig
config
=
new
MessageStoreConfig
();
config
.
setStorePathCommitLog
(
"target/unit_test_store/b/"
);
config
.
setReadOnlyCommitLogStorePaths
(
"target/unit_test_store/a"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/c"
);
MultiPathMappedFileQueue
mappedFileQueue
=
new
MultiPathMappedFileQueue
(
config
,
1024
,
null
,
null
);
mappedFileQueue
.
load
();
assertThat
(
mappedFileQueue
.
mappedFiles
.
size
()).
isEqualTo
(
1024
);
for
(
int
i
=
0
;
i
<
1024
;
i
++)
{
assertThat
(
mappedFileQueue
.
mappedFiles
.
get
(
i
).
getFile
().
getName
())
.
isEqualTo
(
UtilAll
.
offset2FileName
(
1024
*
i
));
}
mappedFileQueue
.
destroy
();
}
@Test
public
void
testUpdatePathsOnline
()
{
final
byte
[]
fixedMsg
=
new
byte
[
1024
];
MessageStoreConfig
config
=
new
MessageStoreConfig
();
config
.
setStorePathCommitLog
(
"target/unit_test_store/a/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/b/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/c/"
);
MappedFileQueue
mappedFileQueue
=
new
MultiPathMappedFileQueue
(
config
,
1024
,
null
,
null
);
String
[]
storePaths
=
config
.
getStorePathCommitLog
().
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
for
(
int
i
=
0
;
i
<
1024
;
i
++)
{
MappedFile
mappedFile
=
mappedFileQueue
.
getLastMappedFile
(
fixedMsg
.
length
*
i
);
assertThat
(
mappedFile
).
isNotNull
();
assertThat
(
mappedFile
.
appendMessage
(
fixedMsg
)).
isTrue
();
int
idx
=
i
%
storePaths
.
length
;
assertThat
(
mappedFile
.
getFileName
().
startsWith
(
storePaths
[
idx
])).
isTrue
();
if
(
i
==
500
)
{
config
.
setStorePathCommitLog
(
"target/unit_test_store/a/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/b/"
);
storePaths
=
config
.
getStorePathCommitLog
().
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
}
}
mappedFileQueue
.
shutdown
(
1000
);
mappedFileQueue
.
destroy
();
}
@Test
public
void
testFullStorePath
()
{
final
byte
[]
fixedMsg
=
new
byte
[
1024
];
Set
<
String
>
fullStorePath
=
new
HashSet
<>();
MessageStoreConfig
config
=
new
MessageStoreConfig
();
config
.
setStorePathCommitLog
(
"target/unit_test_store/a/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/b/"
+
MessageStoreConfig
.
MULTI_PATH_SPLITTER
+
"target/unit_test_store/c/"
);
MappedFileQueue
mappedFileQueue
=
new
MultiPathMappedFileQueue
(
config
,
1024
,
null
,
()
->
fullStorePath
);
String
[]
storePaths
=
config
.
getStorePathCommitLog
().
trim
().
split
(
MessageStoreConfig
.
MULTI_PATH_SPLITTER
);
assertThat
(
storePaths
.
length
).
isEqualTo
(
3
);
MappedFile
mappedFile
=
mappedFileQueue
.
getLastMappedFile
(
0
);
assertThat
(
mappedFile
).
isNotNull
();
assertThat
(
mappedFile
.
appendMessage
(
fixedMsg
)).
isTrue
();
assertThat
(
mappedFile
.
getFileName
().
startsWith
(
storePaths
[
0
])).
isTrue
();
mappedFile
=
mappedFileQueue
.
getLastMappedFile
(
fixedMsg
.
length
);
assertThat
(
mappedFile
.
getFileName
().
startsWith
(
storePaths
[
1
])).
isTrue
();
assertThat
(
mappedFile
.
appendMessage
(
fixedMsg
)).
isTrue
();
mappedFile
=
mappedFileQueue
.
getLastMappedFile
(
fixedMsg
.
length
*
2
);
assertThat
(
mappedFile
.
appendMessage
(
fixedMsg
)).
isTrue
();
assertThat
(
mappedFile
.
getFileName
().
startsWith
(
storePaths
[
2
])).
isTrue
();
fullStorePath
.
add
(
"target/unit_test_store/b/"
);
mappedFile
=
mappedFileQueue
.
getLastMappedFile
(
fixedMsg
.
length
*
3
);
assertThat
(
mappedFile
.
appendMessage
(
fixedMsg
)).
isTrue
();
assertThat
(
mappedFile
.
getFileName
().
startsWith
(
storePaths
[
2
])).
isTrue
();
mappedFile
=
mappedFileQueue
.
getLastMappedFile
(
fixedMsg
.
length
*
4
);
assertThat
(
mappedFile
.
appendMessage
(
fixedMsg
)).
isTrue
();
assertThat
(
mappedFile
.
getFileName
().
startsWith
(
storePaths
[
0
])).
isTrue
();
mappedFileQueue
.
shutdown
(
1000
);
mappedFileQueue
.
destroy
();
}
}
\ No newline at end of file
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录