mirror of
https://github.com/apache/rocketmq-dashboard.git
synced 2025-09-10 19:48:29 +08:00
Compare commits
28 Commits
develop
...
dependabot
Author | SHA1 | Date | |
---|---|---|---|
|
1f4adc3355 | ||
|
ea8834bacd | ||
|
63d9da7429 | ||
|
daa181ccfd | ||
|
f5b9bbb9a7 | ||
|
6de5d51661 | ||
|
4269879d93 | ||
|
0fc7ca904d | ||
|
636ba51bc3 | ||
|
8c5d8747a7 | ||
|
3f02992e13 | ||
|
78b85ebdb6 | ||
|
653332d488 | ||
|
d2da9ca41f | ||
|
6da21f480e | ||
|
e281b74456 | ||
|
529501c007 | ||
|
560b56e9dc | ||
|
8acd94c89a | ||
|
2d8bf001b3 | ||
|
1caeb4ce31 | ||
|
749e6f15f7 | ||
|
2524ca668a | ||
|
6054c057c8 | ||
|
e660c7d874 | ||
|
a5a3659c04 | ||
|
2160e23b91 | ||
|
ef7b97e96a |
@@ -9,3 +9,10 @@ github:
|
|||||||
squash: true
|
squash: true
|
||||||
merge: false
|
merge: false
|
||||||
rebase: false
|
rebase: false
|
||||||
|
|
||||||
|
notifications:
|
||||||
|
commits: commits@rocketmq.apache.org
|
||||||
|
issues: commits@rocketmq.apache.org
|
||||||
|
pullrequests: commits@rocketmq.apache.org
|
||||||
|
jobs: commits@rocketmq.apache.org
|
||||||
|
discussions: dev@rocketmq.apache.org
|
||||||
|
@@ -20,3 +20,5 @@ script:
|
|||||||
# - travis_retry mvn -B package findbugs:findbugs coveralls:report
|
# - travis_retry mvn -B package findbugs:findbugs coveralls:report
|
||||||
#after_success:
|
#after_success:
|
||||||
# - mvn sonar:sonar
|
# - mvn sonar:sonar
|
||||||
|
after_success:
|
||||||
|
- bash <(curl -s https://codecov.io/bash) -t 996e4d8e-8a00-49f2-b649-38c03716d690 || echo 'Codecov failed to upload'
|
||||||
|
2
LICENSE
2
LICENSE
@@ -15,7 +15,7 @@
|
|||||||
"Legal Entity" shall mean the union of the acting entity and all
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
other entities that control, are controlled by, or are under common
|
other entities that control, are controlled by, or are under common
|
||||||
control with that entity. For the purposes of this definition,
|
control with that entity. For the purposes of this definition,
|
||||||
"control" means (properties) the power, direct or indirect, to cause the
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
direction or management of such entity, whether by contract or
|
direction or management of such entity, whether by contract or
|
||||||
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
2
NOTICE
2
NOTICE
@@ -1,5 +1,5 @@
|
|||||||
Apache RocketMQ
|
Apache RocketMQ
|
||||||
Copyright 2016-2021 The Apache Software Foundation
|
Copyright 2016-2022 The Apache Software Foundation
|
||||||
|
|
||||||
This product includes software developed at
|
This product includes software developed at
|
||||||
The Apache Software Foundation (http://www.apache.org/).
|
The Apache Software Foundation (http://www.apache.org/).
|
||||||
|
@@ -1,5 +1,7 @@
|
|||||||
## [Apache RocketMQ](https://github.com/apache/rocketmq) Dashboard [](https://travis-ci.com/github/apache/rocketmq-dashboard) [](https://coveralls.io/github/apache/rocketmq-dashboard?branch=master)
|
## [Apache RocketMQ](https://github.com/apache/rocketmq) Dashboard
|
||||||
|
[](https://travis-ci.com/github/apache/rocketmq-dashboard) [](https://coveralls.io/github/apache/rocketmq-dashboard?branch=master)
|
||||||
[](https://www.apache.org/licenses/LICENSE-2.0.html)
|
[](https://www.apache.org/licenses/LICENSE-2.0.html)
|
||||||
|
[](https://codecov.io/gh/apache/rocketmq-dashboard)
|
||||||
[](http://isitmaintained.com/project/apache/rocketmq-dashboard "Average time to resolve an issue")
|
[](http://isitmaintained.com/project/apache/rocketmq-dashboard "Average time to resolve an issue")
|
||||||
[](http://isitmaintained.com/project/apache/rocketmq-dashboard "Percentage of issues still open")
|
[](http://isitmaintained.com/project/apache/rocketmq-dashboard "Percentage of issues still open")
|
||||||
[](https://twitter.com/intent/follow?screen_name=ApacheRocketMQ)
|
[](https://twitter.com/intent/follow?screen_name=ApacheRocketMQ)
|
||||||
|
17388
frontend/package-lock.json
generated
17388
frontend/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -9,7 +9,7 @@
|
|||||||
"react": "^17.0.2",
|
"react": "^17.0.2",
|
||||||
"react-dom": "^17.0.2",
|
"react-dom": "^17.0.2",
|
||||||
"react-json-view": "^1.21.3",
|
"react-json-view": "^1.21.3",
|
||||||
"react-scripts": "4.0.3",
|
"react-scripts": "5.0.1",
|
||||||
"web-vitals": "^1.0.1"
|
"web-vitals": "^1.0.1"
|
||||||
},
|
},
|
||||||
"proxy": "http://localhost:8080",
|
"proxy": "http://localhost:8080",
|
||||||
|
11581
frontend/yarn.lock
11581
frontend/yarn.lock
File diff suppressed because it is too large
Load Diff
40
pom.xml
40
pom.xml
@@ -90,18 +90,24 @@
|
|||||||
<commons-lang.version>2.6</commons-lang.version>
|
<commons-lang.version>2.6</commons-lang.version>
|
||||||
<commons-io.version>2.4</commons-io.version>
|
<commons-io.version>2.4</commons-io.version>
|
||||||
<commons-cli.version>1.2</commons-cli.version>
|
<commons-cli.version>1.2</commons-cli.version>
|
||||||
<rocketmq.version>4.9.0</rocketmq.version>
|
<commons-collections.version>3.2.2</commons-collections.version>
|
||||||
|
<rocketmq.version>4.9.3</rocketmq.version>
|
||||||
<surefire.version>2.19.1</surefire.version>
|
<surefire.version>2.19.1</surefire.version>
|
||||||
<aspectj.version>1.9.6</aspectj.version>
|
<aspectj.version>1.9.6</aspectj.version>
|
||||||
<lombok.version>1.18.12</lombok.version>
|
<lombok.version>1.18.12</lombok.version>
|
||||||
<main.basedir>${basedir}/../..</main.basedir>
|
<main.basedir>${basedir}/../..</main.basedir>
|
||||||
<docker.image.prefix>apacherocketmq</docker.image.prefix>
|
<docker.image.prefix>apacherocketmq</docker.image.prefix>
|
||||||
<spring.boot.version>2.2.2.RELEASE</spring.boot.version>
|
<spring.boot.version>2.6.0</spring.boot.version>
|
||||||
<mockito-inline.version>3.3.3</mockito-inline.version>
|
<mockito-inline.version>3.3.3</mockito-inline.version>
|
||||||
<jaxb-api.version>2.3.1</jaxb-api.version>
|
<jaxb-api.version>2.3.1</jaxb-api.version>
|
||||||
<commons-pool2.version>2.4.3</commons-pool2.version>
|
<commons-pool2.version>2.4.3</commons-pool2.version>
|
||||||
<easyexcel.version>2.2.10</easyexcel.version>
|
<easyexcel.version>2.2.10</easyexcel.version>
|
||||||
<asm.version>4.2</asm.version>
|
<asm.version>4.2</asm.version>
|
||||||
|
<junit.version>4.12</junit.version>
|
||||||
|
<snakeyaml.version>1.30</snakeyaml.version>
|
||||||
|
<cglib.version>2.2.2</cglib.version>
|
||||||
|
<joor.version>0.9.6</joor.version>
|
||||||
|
<bcpkix-jdk15on.version>1.68</bcpkix-jdk15on.version>
|
||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
@@ -125,11 +131,22 @@
|
|||||||
<artifactId>spring-boot-starter-test</artifactId>
|
<artifactId>spring-boot-starter-test</artifactId>
|
||||||
<version>${spring.boot.version}</version>
|
<version>${spring.boot.version}</version>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
|
<exclusions>
|
||||||
|
<exclusion>
|
||||||
|
<groupId>org.mockito</groupId>
|
||||||
|
<artifactId>mockito-core</artifactId>
|
||||||
|
</exclusion>
|
||||||
|
</exclusions>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot-starter-validation</artifactId>
|
||||||
|
<version>${spring.boot.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>commons-collections</groupId>
|
<groupId>commons-collections</groupId>
|
||||||
<artifactId>commons-collections</artifactId>
|
<artifactId>commons-collections</artifactId>
|
||||||
<version>3.2.2</version>
|
<version>${commons-collections.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.apache.rocketmq</groupId>
|
<groupId>org.apache.rocketmq</groupId>
|
||||||
@@ -203,17 +220,17 @@
|
|||||||
<dependency>
|
<dependency>
|
||||||
<groupId>cglib</groupId>
|
<groupId>cglib</groupId>
|
||||||
<artifactId>cglib</artifactId>
|
<artifactId>cglib</artifactId>
|
||||||
<version>2.2.2</version>
|
<version>${cglib.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.jooq</groupId>
|
<groupId>org.jooq</groupId>
|
||||||
<artifactId>joor</artifactId>
|
<artifactId>joor</artifactId>
|
||||||
<version>0.9.6</version>
|
<version>${joor.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.bouncycastle</groupId>
|
<groupId>org.bouncycastle</groupId>
|
||||||
<artifactId>bcpkix-jdk15on</artifactId>
|
<artifactId>bcpkix-jdk15on</artifactId>
|
||||||
<version>1.68</version>
|
<version>${bcpkix-jdk15on.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>javax.xml.bind</groupId>
|
<groupId>javax.xml.bind</groupId>
|
||||||
@@ -246,6 +263,17 @@
|
|||||||
<artifactId>asm</artifactId>
|
<artifactId>asm</artifactId>
|
||||||
<version>${asm.version}</version>
|
<version>${asm.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>junit</groupId>
|
||||||
|
<artifactId>junit</artifactId>
|
||||||
|
<version>${junit.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.yaml</groupId>
|
||||||
|
<artifactId>snakeyaml</artifactId>
|
||||||
|
<version>${snakeyaml.version}</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
</dependencies>
|
</dependencies>
|
||||||
<build>
|
<build>
|
||||||
|
@@ -16,6 +16,7 @@
|
|||||||
*/
|
*/
|
||||||
package org.apache.rocketmq.dashboard.admin;
|
package org.apache.rocketmq.dashboard.admin;
|
||||||
|
|
||||||
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.apache.rocketmq.acl.common.AclClientRPCHook;
|
import org.apache.rocketmq.acl.common.AclClientRPCHook;
|
||||||
@@ -33,6 +34,8 @@ public class MQAdminFactory {
|
|||||||
this.rmqConfigure = rmqConfigure;
|
this.rmqConfigure = rmqConfigure;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private final AtomicLong adminIndex = new AtomicLong(0);
|
||||||
|
|
||||||
public MQAdminExt getInstance() throws Exception {
|
public MQAdminExt getInstance() throws Exception {
|
||||||
RPCHook rpcHook = null;
|
RPCHook rpcHook = null;
|
||||||
final String accessKey = rmqConfigure.getAccessKey();
|
final String accessKey = rmqConfigure.getAccessKey();
|
||||||
@@ -47,6 +50,7 @@ public class MQAdminFactory {
|
|||||||
} else {
|
} else {
|
||||||
mqAdminExt = new DefaultMQAdminExt(rpcHook, rmqConfigure.getTimeoutMillis());
|
mqAdminExt = new DefaultMQAdminExt(rpcHook, rmqConfigure.getTimeoutMillis());
|
||||||
}
|
}
|
||||||
|
mqAdminExt.setAdminExtGroup(mqAdminExt.getAdminExtGroup() + "_" + adminIndex.getAndIncrement());
|
||||||
mqAdminExt.setVipChannelEnabled(Boolean.parseBoolean(rmqConfigure.getIsVIPChannel()));
|
mqAdminExt.setVipChannelEnabled(Boolean.parseBoolean(rmqConfigure.getIsVIPChannel()));
|
||||||
mqAdminExt.setUseTLS(rmqConfigure.isUseTLS());
|
mqAdminExt.setUseTLS(rmqConfigure.isUseTLS());
|
||||||
mqAdminExt.setInstanceName(Long.toString(System.currentTimeMillis()));
|
mqAdminExt.setInstanceName(Long.toString(System.currentTimeMillis()));
|
||||||
|
@@ -50,17 +50,19 @@ public class AuthWebMVCConfigurerAdapter extends WebMvcConfigurerAdapter {
|
|||||||
public void addInterceptors(InterceptorRegistry registry) {
|
public void addInterceptors(InterceptorRegistry registry) {
|
||||||
if (configure.isLoginRequired()) {
|
if (configure.isLoginRequired()) {
|
||||||
registry.addInterceptor(authInterceptor).addPathPatterns(
|
registry.addInterceptor(authInterceptor).addPathPatterns(
|
||||||
"/cluster/**",
|
"/cluster/**",
|
||||||
"/consumer/**",
|
"/consumer/**",
|
||||||
"/dashboard/**",
|
"/dashboard/**",
|
||||||
"/message/**",
|
"/dlqMessage/**",
|
||||||
"/messageTrace/**",
|
"/message/**",
|
||||||
"/monitor/**",
|
"/messageTrace/**",
|
||||||
"/rocketmq/**",
|
"/monitor/**",
|
||||||
"/ops/**",
|
"/rocketmq/**",
|
||||||
"/producer/**",
|
"/ops/**",
|
||||||
"/test/**",
|
"/producer/**",
|
||||||
"/topic/**");
|
"/test/**",
|
||||||
|
"/topic/**",
|
||||||
|
"/acl/**");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -75,9 +77,9 @@ public class AuthWebMVCConfigurerAdapter extends WebMvcConfigurerAdapter {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object resolveArgument(MethodParameter methodParameter, ModelAndViewContainer modelAndViewContainer,
|
public Object resolveArgument(MethodParameter methodParameter, ModelAndViewContainer modelAndViewContainer,
|
||||||
NativeWebRequest nativeWebRequest, WebDataBinderFactory webDataBinderFactory) throws Exception {
|
NativeWebRequest nativeWebRequest, WebDataBinderFactory webDataBinderFactory) throws Exception {
|
||||||
UserInfo userInfo = (UserInfo) WebUtil.getValueFromSession((HttpServletRequest) nativeWebRequest.getNativeRequest(),
|
UserInfo userInfo = (UserInfo) WebUtil.getValueFromSession((HttpServletRequest) nativeWebRequest.getNativeRequest(),
|
||||||
UserInfo.USER_INFO);
|
UserInfo.USER_INFO);
|
||||||
if (userInfo != null) {
|
if (userInfo != null) {
|
||||||
return userInfo;
|
return userInfo;
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,60 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.config;
|
||||||
|
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.LinkedBlockingDeque;
|
||||||
|
import java.util.concurrent.ThreadFactory;
|
||||||
|
import java.util.concurrent.ThreadPoolExecutor;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
import lombok.Data;
|
||||||
|
import org.springframework.boot.context.properties.ConfigurationProperties;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
@ConfigurationProperties(prefix = "threadpool.config")
|
||||||
|
@Data
|
||||||
|
public class CollectExecutorConfig {
|
||||||
|
private int coreSize = 20;
|
||||||
|
private int maxSize = 20;
|
||||||
|
private long keepAliveTime = 3000L;
|
||||||
|
private int queueSize = 1000;
|
||||||
|
|
||||||
|
@Bean(name = "collectExecutor")
|
||||||
|
public ExecutorService collectExecutor(CollectExecutorConfig collectExecutorConfig) {
|
||||||
|
ExecutorService collectExecutor = new ThreadPoolExecutor(
|
||||||
|
collectExecutorConfig.getCoreSize(),
|
||||||
|
collectExecutorConfig.getMaxSize(),
|
||||||
|
collectExecutorConfig.getKeepAliveTime(),
|
||||||
|
TimeUnit.MILLISECONDS,
|
||||||
|
new LinkedBlockingDeque<>(collectExecutorConfig.getQueueSize()),
|
||||||
|
new ThreadFactory() {
|
||||||
|
private final AtomicLong threadIndex = new AtomicLong(0);
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Thread newThread(Runnable r) {
|
||||||
|
return new Thread(r, "collectTopicThread_" + this.threadIndex.incrementAndGet());
|
||||||
|
}
|
||||||
|
},
|
||||||
|
new ThreadPoolExecutor.DiscardOldestPolicy()
|
||||||
|
);
|
||||||
|
return collectExecutor;
|
||||||
|
}
|
||||||
|
}
|
@@ -16,6 +16,8 @@
|
|||||||
*/
|
*/
|
||||||
package org.apache.rocketmq.dashboard.config;
|
package org.apache.rocketmq.dashboard.config;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import org.apache.commons.collections.CollectionUtils;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.apache.rocketmq.common.MixAll;
|
import org.apache.rocketmq.common.MixAll;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
@@ -29,6 +31,7 @@ import org.springframework.context.annotation.Configuration;
|
|||||||
import org.springframework.http.HttpStatus;
|
import org.springframework.http.HttpStatus;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import static org.apache.rocketmq.client.ClientConfig.SEND_MESSAGE_WITH_VIP_CHANNEL_PROPERTY;
|
import static org.apache.rocketmq.client.ClientConfig.SEND_MESSAGE_WITH_VIP_CHANNEL_PROPERTY;
|
||||||
|
|
||||||
@@ -57,6 +60,8 @@ public class RMQConfigure {
|
|||||||
|
|
||||||
private Long timeoutMillis;
|
private Long timeoutMillis;
|
||||||
|
|
||||||
|
private List<String> namesrvAddrs = new ArrayList<>();
|
||||||
|
|
||||||
public String getAccessKey() {
|
public String getAccessKey() {
|
||||||
return accessKey;
|
return accessKey;
|
||||||
}
|
}
|
||||||
@@ -77,6 +82,17 @@ public class RMQConfigure {
|
|||||||
return namesrvAddr;
|
return namesrvAddr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public List<String> getNamesrvAddrs() {
|
||||||
|
return namesrvAddrs;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setNamesrvAddrs(List<String> namesrvAddrs) {
|
||||||
|
this.namesrvAddrs = namesrvAddrs;
|
||||||
|
if (CollectionUtils.isNotEmpty(namesrvAddrs)) {
|
||||||
|
this.setNamesrvAddr(namesrvAddrs.get(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void setNamesrvAddr(String namesrvAddr) {
|
public void setNamesrvAddr(String namesrvAddr) {
|
||||||
if (StringUtils.isNotBlank(namesrvAddr)) {
|
if (StringUtils.isNotBlank(namesrvAddr)) {
|
||||||
this.namesrvAddr = namesrvAddr;
|
this.namesrvAddr = namesrvAddr;
|
||||||
|
@@ -0,0 +1,146 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.controller;
|
||||||
|
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
|
import java.util.List;
|
||||||
|
import javax.annotation.Resource;
|
||||||
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
import org.apache.commons.collections.CollectionUtils;
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
|
import org.apache.rocketmq.common.AclConfig;
|
||||||
|
import org.apache.rocketmq.common.PlainAccessConfig;
|
||||||
|
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
||||||
|
import org.apache.rocketmq.dashboard.model.User;
|
||||||
|
import org.apache.rocketmq.dashboard.model.UserInfo;
|
||||||
|
import org.apache.rocketmq.dashboard.model.request.AclRequest;
|
||||||
|
import org.apache.rocketmq.dashboard.permisssion.Permission;
|
||||||
|
import org.apache.rocketmq.dashboard.service.AclService;
|
||||||
|
import org.apache.rocketmq.dashboard.support.JsonResult;
|
||||||
|
import org.apache.rocketmq.dashboard.util.WebUtil;
|
||||||
|
import org.springframework.web.bind.annotation.DeleteMapping;
|
||||||
|
import org.springframework.web.bind.annotation.GetMapping;
|
||||||
|
import org.springframework.web.bind.annotation.PostMapping;
|
||||||
|
import org.springframework.web.bind.annotation.RequestBody;
|
||||||
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
|
import org.springframework.web.bind.annotation.RequestParam;
|
||||||
|
import org.springframework.web.bind.annotation.RestController;
|
||||||
|
|
||||||
|
@RestController
|
||||||
|
@RequestMapping("/acl")
|
||||||
|
@Permission
|
||||||
|
public class AclController {
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
private AclService aclService;
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
private RMQConfigure configure;
|
||||||
|
|
||||||
|
@GetMapping("/enable.query")
|
||||||
|
public Object isEnableAcl() {
|
||||||
|
return new JsonResult<>(configure.isACLEnabled());
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping("/config.query")
|
||||||
|
public AclConfig getAclConfig(HttpServletRequest request) {
|
||||||
|
if (!configure.isLoginRequired()) {
|
||||||
|
return aclService.getAclConfig(false);
|
||||||
|
}
|
||||||
|
UserInfo userInfo = (UserInfo) WebUtil.getValueFromSession(request, WebUtil.USER_INFO);
|
||||||
|
// if user info is null but reach here, must exclude secret key for safety.
|
||||||
|
return aclService.getAclConfig(userInfo == null || userInfo.getUser().getType() != User.ADMIN);
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/add.do")
|
||||||
|
public Object addAclConfig(@RequestBody PlainAccessConfig config) {
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(config.getAccessKey()), "accessKey is null");
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(config.getSecretKey()), "secretKey is null");
|
||||||
|
aclService.addAclConfig(config);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/delete.do")
|
||||||
|
public Object deleteAclConfig(@RequestBody PlainAccessConfig config) {
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(config.getAccessKey()), "accessKey is null");
|
||||||
|
aclService.deleteAclConfig(config);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/update.do")
|
||||||
|
public Object updateAclConfig(@RequestBody PlainAccessConfig config) {
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(config.getSecretKey()), "secretKey is null");
|
||||||
|
aclService.updateAclConfig(config);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/topic/add.do")
|
||||||
|
public Object addAclTopicConfig(@RequestBody AclRequest request) {
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(request.getConfig().getAccessKey()), "accessKey is null");
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(request.getConfig().getSecretKey()), "secretKey is null");
|
||||||
|
Preconditions.checkArgument(CollectionUtils.isNotEmpty(request.getConfig().getTopicPerms()), "topic perms is null");
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(request.getTopicPerm()), "topic perm is null");
|
||||||
|
aclService.addOrUpdateAclTopicConfig(request);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/group/add.do")
|
||||||
|
public Object addAclGroupConfig(@RequestBody AclRequest request) {
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(request.getConfig().getAccessKey()), "accessKey is null");
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(request.getConfig().getSecretKey()), "secretKey is null");
|
||||||
|
Preconditions.checkArgument(CollectionUtils.isNotEmpty(request.getConfig().getGroupPerms()), "group perms is null");
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(request.getGroupPerm()), "group perm is null");
|
||||||
|
aclService.addOrUpdateAclGroupConfig(request);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/perm/delete.do")
|
||||||
|
public Object deletePermConfig(@RequestBody AclRequest request) {
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(request.getConfig().getAccessKey()), "accessKey is null");
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(request.getConfig().getSecretKey()), "secretKey is null");
|
||||||
|
aclService.deletePermConfig(request);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/sync.do")
|
||||||
|
public Object syncConfig(@RequestBody PlainAccessConfig config) {
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(config.getAccessKey()), "accessKey is null");
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(config.getSecretKey()), "secretKey is null");
|
||||||
|
aclService.syncData(config);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/white/list/add.do")
|
||||||
|
public Object addWhiteList(@RequestBody List<String> whiteList) {
|
||||||
|
Preconditions.checkArgument(CollectionUtils.isNotEmpty(whiteList), "white list is null");
|
||||||
|
aclService.addWhiteList(whiteList);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@DeleteMapping("/white/list/delete.do")
|
||||||
|
public Object deleteWhiteAddr(@RequestParam String request) {
|
||||||
|
aclService.deleteWhiteAddr(request);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/white/list/sync.do")
|
||||||
|
public Object synchronizeWhiteList(@RequestBody List<String> whiteList) {
|
||||||
|
Preconditions.checkArgument(CollectionUtils.isNotEmpty(whiteList), "white list is null");
|
||||||
|
aclService.synchronizeWhiteList(whiteList);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
@@ -51,7 +51,7 @@ public class DashboardController {
|
|||||||
return dashboardService.queryTopicData(date,topicName);
|
return dashboardService.queryTopicData(date,topicName);
|
||||||
}
|
}
|
||||||
|
|
||||||
@RequestMapping(value = "/topicCurrent", method = RequestMethod.GET)
|
@RequestMapping(value = "/topicCurrent.query", method = RequestMethod.GET)
|
||||||
@ResponseBody
|
@ResponseBody
|
||||||
public Object topicCurrent() {
|
public Object topicCurrent() {
|
||||||
return dashboardService.queryTopicCurrentData();
|
return dashboardService.queryTopicCurrentData();
|
||||||
|
@@ -17,12 +17,16 @@
|
|||||||
package org.apache.rocketmq.dashboard.controller;
|
package org.apache.rocketmq.dashboard.controller;
|
||||||
|
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
import javax.servlet.http.HttpServletResponse;
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.apache.rocketmq.common.MixAll;
|
import org.apache.rocketmq.common.MixAll;
|
||||||
import org.apache.rocketmq.common.message.MessageExt;
|
import org.apache.rocketmq.common.message.MessageExt;
|
||||||
import org.apache.rocketmq.dashboard.exception.ServiceException;
|
import org.apache.rocketmq.dashboard.exception.ServiceException;
|
||||||
import org.apache.rocketmq.dashboard.model.DlqMessageExcelModel;
|
import org.apache.rocketmq.dashboard.model.DlqMessageExcelModel;
|
||||||
|
import org.apache.rocketmq.dashboard.model.DlqMessageRequest;
|
||||||
import org.apache.rocketmq.dashboard.model.request.MessageQuery;
|
import org.apache.rocketmq.dashboard.model.request.MessageQuery;
|
||||||
import org.apache.rocketmq.dashboard.permisssion.Permission;
|
import org.apache.rocketmq.dashboard.permisssion.Permission;
|
||||||
import org.apache.rocketmq.dashboard.service.DlqMessageService;
|
import org.apache.rocketmq.dashboard.service.DlqMessageService;
|
||||||
@@ -30,6 +34,7 @@ import org.apache.rocketmq.dashboard.util.ExcelUtil;
|
|||||||
import org.apache.rocketmq.tools.admin.MQAdminExt;
|
import org.apache.rocketmq.tools.admin.MQAdminExt;
|
||||||
import org.springframework.stereotype.Controller;
|
import org.springframework.stereotype.Controller;
|
||||||
import org.springframework.web.bind.annotation.GetMapping;
|
import org.springframework.web.bind.annotation.GetMapping;
|
||||||
|
import org.springframework.web.bind.annotation.PostMapping;
|
||||||
import org.springframework.web.bind.annotation.RequestBody;
|
import org.springframework.web.bind.annotation.RequestBody;
|
||||||
import org.springframework.web.bind.annotation.RequestMapping;
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
import org.springframework.web.bind.annotation.RequestMethod;
|
import org.springframework.web.bind.annotation.RequestMethod;
|
||||||
@@ -39,6 +44,7 @@ import org.springframework.web.bind.annotation.ResponseBody;
|
|||||||
@Controller
|
@Controller
|
||||||
@RequestMapping("/dlqMessage")
|
@RequestMapping("/dlqMessage")
|
||||||
@Permission
|
@Permission
|
||||||
|
@Slf4j
|
||||||
public class DlqMessageController {
|
public class DlqMessageController {
|
||||||
|
|
||||||
@Resource
|
@Resource
|
||||||
@@ -70,4 +76,33 @@ public class DlqMessageController {
|
|||||||
throw new ServiceException(-1, String.format("export dlq message failed!"));
|
throw new ServiceException(-1, String.format("export dlq message failed!"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@PostMapping(value = "/batchResendDlqMessage.do")
|
||||||
|
@ResponseBody
|
||||||
|
public Object batchResendDlqMessage(@RequestBody List<DlqMessageRequest> dlqMessages) {
|
||||||
|
return dlqMessageService.batchResendDlqMessage(dlqMessages);
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping(value = "/batchExportDlqMessage.do")
|
||||||
|
public void batchExportDlqMessage(HttpServletResponse response, @RequestBody List<DlqMessageRequest> dlqMessages) {
|
||||||
|
List<DlqMessageExcelModel> dlqMessageExcelModelList = new ArrayList<>(dlqMessages.size());
|
||||||
|
for (DlqMessageRequest dlqMessage : dlqMessages) {
|
||||||
|
DlqMessageExcelModel excelModel = new DlqMessageExcelModel();
|
||||||
|
try {
|
||||||
|
String topic = MixAll.DLQ_GROUP_TOPIC_PREFIX + dlqMessage.getConsumerGroup();
|
||||||
|
MessageExt messageExt = mqAdminExt.viewMessage(topic, dlqMessage.getMsgId());
|
||||||
|
excelModel = new DlqMessageExcelModel(messageExt);
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("Failed to query message by Id:{}", dlqMessage.getMsgId(), e);
|
||||||
|
excelModel.setMsgId(dlqMessage.getMsgId());
|
||||||
|
excelModel.setException(e.getMessage());
|
||||||
|
}
|
||||||
|
dlqMessageExcelModelList.add(excelModel);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
ExcelUtil.writeExcel(response, dlqMessageExcelModelList, "dlqs", "dlqs", DlqMessageExcelModel.class);
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new ServiceException(-1, String.format("export dlq message failed!"));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -32,7 +32,7 @@ public class NamesvrController {
|
|||||||
@Resource
|
@Resource
|
||||||
private OpsService opsService;
|
private OpsService opsService;
|
||||||
|
|
||||||
@RequestMapping(value = "/nsaddr", method = RequestMethod.GET)
|
@RequestMapping(value = "/nsaddr.query", method = RequestMethod.GET)
|
||||||
@ResponseBody
|
@ResponseBody
|
||||||
@OriginalControllerReturnValue
|
@OriginalControllerReturnValue
|
||||||
public Object nsaddr() {
|
public Object nsaddr() {
|
||||||
|
@@ -16,7 +16,9 @@
|
|||||||
*/
|
*/
|
||||||
package org.apache.rocketmq.dashboard.controller;
|
package org.apache.rocketmq.dashboard.controller;
|
||||||
|
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.apache.rocketmq.dashboard.permisssion.Permission;
|
import org.apache.rocketmq.dashboard.permisssion.Permission;
|
||||||
import org.apache.rocketmq.dashboard.service.OpsService;
|
import org.apache.rocketmq.dashboard.service.OpsService;
|
||||||
import org.springframework.stereotype.Controller;
|
import org.springframework.stereotype.Controller;
|
||||||
@@ -46,6 +48,15 @@ public class OpsController {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@RequestMapping(value = "/addNameSvrAddr.do", method = RequestMethod.POST)
|
||||||
|
@ResponseBody
|
||||||
|
public Object addNameSvrAddr(@RequestParam String newNamesrvAddr) {
|
||||||
|
Preconditions.checkArgument(StringUtils.isNotEmpty(newNamesrvAddr),
|
||||||
|
"namesrvAddr can not be blank");
|
||||||
|
opsService.addNameSvrAddr(newNamesrvAddr);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
@RequestMapping(value = "/updateIsVIPChannel.do", method = RequestMethod.POST)
|
@RequestMapping(value = "/updateIsVIPChannel.do", method = RequestMethod.POST)
|
||||||
@ResponseBody
|
@ResponseBody
|
||||||
public Object updateIsVIPChannel(@RequestParam String useVIPChannel) {
|
public Object updateIsVIPChannel(@RequestParam String useVIPChannel) {
|
||||||
@@ -53,14 +64,12 @@ public class OpsController {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@RequestMapping(value = "/rocketMqStatus.query", method = RequestMethod.GET)
|
@RequestMapping(value = "/rocketMqStatus.query", method = RequestMethod.GET)
|
||||||
@ResponseBody
|
@ResponseBody
|
||||||
public Object clusterStatus() {
|
public Object clusterStatus() {
|
||||||
return opsService.rocketMqStatusCheck();
|
return opsService.rocketMqStatusCheck();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@RequestMapping(value = "/updateUseTLS.do", method = RequestMethod.POST)
|
@RequestMapping(value = "/updateUseTLS.do", method = RequestMethod.POST)
|
||||||
@ResponseBody
|
@ResponseBody
|
||||||
public Object updateUseTLS(@RequestParam String useTLS) {
|
public Object updateUseTLS(@RequestParam String useTLS) {
|
||||||
|
@@ -25,9 +25,11 @@ import com.google.common.base.Charsets;
|
|||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
|
import lombok.NoArgsConstructor;
|
||||||
import org.apache.rocketmq.common.message.MessageExt;
|
import org.apache.rocketmq.common.message.MessageExt;
|
||||||
|
|
||||||
@Data
|
@Data
|
||||||
|
@NoArgsConstructor
|
||||||
public class DlqMessageExcelModel extends BaseRowModel implements Serializable {
|
public class DlqMessageExcelModel extends BaseRowModel implements Serializable {
|
||||||
|
|
||||||
@ExcelProperty(value = "topic", index = 0)
|
@ExcelProperty(value = "topic", index = 0)
|
||||||
@@ -66,6 +68,10 @@ public class DlqMessageExcelModel extends BaseRowModel implements Serializable {
|
|||||||
@ColumnWidth(value = 15)
|
@ColumnWidth(value = 15)
|
||||||
private int bodyCRC;
|
private int bodyCRC;
|
||||||
|
|
||||||
|
@ExcelProperty(value = "exception", index = 9)
|
||||||
|
@ColumnWidth(value = 30)
|
||||||
|
private String exception;
|
||||||
|
|
||||||
public DlqMessageExcelModel(MessageExt messageExt) {
|
public DlqMessageExcelModel(MessageExt messageExt) {
|
||||||
this.topic = messageExt.getTopic();
|
this.topic = messageExt.getTopic();
|
||||||
this.msgId = messageExt.getMsgId();
|
this.msgId = messageExt.getMsgId();
|
||||||
|
@@ -0,0 +1,31 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.model;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
public class DlqMessageRequest {
|
||||||
|
|
||||||
|
private String topicName;
|
||||||
|
|
||||||
|
private String consumerGroup;
|
||||||
|
|
||||||
|
private String msgId;
|
||||||
|
|
||||||
|
private String clientId;
|
||||||
|
}
|
@@ -0,0 +1,34 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.model;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.CMResult;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.ConsumeMessageDirectlyResult;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
public class DlqMessageResendResult {
|
||||||
|
private CMResult consumeResult;
|
||||||
|
private String remark;
|
||||||
|
private String msgId;
|
||||||
|
|
||||||
|
public DlqMessageResendResult(ConsumeMessageDirectlyResult consumeMessageDirectlyResult, String msgId) {
|
||||||
|
this.consumeResult = consumeMessageDirectlyResult.getConsumeResult();
|
||||||
|
this.remark = consumeMessageDirectlyResult.getRemark();
|
||||||
|
this.msgId = msgId;
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,30 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.model.request;
|
||||||
|
|
||||||
|
import lombok.Data;
|
||||||
|
import org.apache.rocketmq.common.PlainAccessConfig;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
public class AclRequest {
|
||||||
|
|
||||||
|
private PlainAccessConfig config;
|
||||||
|
|
||||||
|
private String topicPerm;
|
||||||
|
|
||||||
|
private String groupPerm;
|
||||||
|
}
|
@@ -0,0 +1,47 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.service;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import org.apache.rocketmq.common.AclConfig;
|
||||||
|
import org.apache.rocketmq.common.PlainAccessConfig;
|
||||||
|
import org.apache.rocketmq.dashboard.model.request.AclRequest;
|
||||||
|
|
||||||
|
public interface AclService {
|
||||||
|
|
||||||
|
AclConfig getAclConfig(boolean excludeSecretKey);
|
||||||
|
|
||||||
|
void addAclConfig(PlainAccessConfig config);
|
||||||
|
|
||||||
|
void deleteAclConfig(PlainAccessConfig config);
|
||||||
|
|
||||||
|
void updateAclConfig(PlainAccessConfig config);
|
||||||
|
|
||||||
|
void addOrUpdateAclTopicConfig(AclRequest request);
|
||||||
|
|
||||||
|
void addOrUpdateAclGroupConfig(AclRequest request);
|
||||||
|
|
||||||
|
void deletePermConfig(AclRequest request);
|
||||||
|
|
||||||
|
void syncData(PlainAccessConfig config);
|
||||||
|
|
||||||
|
void addWhiteList(List<String> whiteList);
|
||||||
|
|
||||||
|
void deleteWhiteAddr(String addr);
|
||||||
|
|
||||||
|
void synchronizeWhiteList(List<String> whiteList);
|
||||||
|
}
|
@@ -17,10 +17,15 @@
|
|||||||
|
|
||||||
package org.apache.rocketmq.dashboard.service;
|
package org.apache.rocketmq.dashboard.service;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import org.apache.rocketmq.dashboard.model.DlqMessageResendResult;
|
||||||
|
import org.apache.rocketmq.dashboard.model.DlqMessageRequest;
|
||||||
import org.apache.rocketmq.dashboard.model.MessagePage;
|
import org.apache.rocketmq.dashboard.model.MessagePage;
|
||||||
import org.apache.rocketmq.dashboard.model.request.MessageQuery;
|
import org.apache.rocketmq.dashboard.model.request.MessageQuery;
|
||||||
|
|
||||||
public interface DlqMessageService {
|
public interface DlqMessageService {
|
||||||
|
|
||||||
MessagePage queryDlqMessageByPage(MessageQuery query);
|
MessagePage queryDlqMessageByPage(MessageQuery query);
|
||||||
|
|
||||||
|
List<DlqMessageResendResult> batchResendDlqMessage(List<DlqMessageRequest> dlqMessages);
|
||||||
}
|
}
|
||||||
|
@@ -31,4 +31,6 @@ public interface OpsService {
|
|||||||
boolean updateIsVIPChannel(String useVIPChannel);
|
boolean updateIsVIPChannel(String useVIPChannel);
|
||||||
|
|
||||||
boolean updateUseTLS(boolean useTLS);
|
boolean updateUseTLS(boolean useTLS);
|
||||||
|
|
||||||
|
void addNameSvrAddr(String namesrvAddr);
|
||||||
}
|
}
|
||||||
|
@@ -91,29 +91,34 @@ public class MQAdminExtImpl implements MQAdminExt {
|
|||||||
MQAdminInstance.threadLocalMQAdminExt().createAndUpdateTopicConfig(addr, config);
|
MQAdminInstance.threadLocalMQAdminExt().createAndUpdateTopicConfig(addr, config);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void createAndUpdatePlainAccessConfig(String addr,
|
@Override
|
||||||
|
public void createAndUpdatePlainAccessConfig(String addr,
|
||||||
PlainAccessConfig plainAccessConfig) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
PlainAccessConfig plainAccessConfig) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
||||||
|
MQAdminInstance.threadLocalMQAdminExt().createAndUpdatePlainAccessConfig(addr, plainAccessConfig);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void deletePlainAccessConfig(String addr,
|
@Override
|
||||||
|
public void deletePlainAccessConfig(String addr,
|
||||||
String accessKey) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
String accessKey) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
||||||
|
MQAdminInstance.threadLocalMQAdminExt().deletePlainAccessConfig(addr, accessKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void updateGlobalWhiteAddrConfig(String addr,
|
@Override
|
||||||
|
public void updateGlobalWhiteAddrConfig(String addr,
|
||||||
String globalWhiteAddrs) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
String globalWhiteAddrs) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
||||||
|
MQAdminInstance.threadLocalMQAdminExt().updateGlobalWhiteAddrConfig(addr, globalWhiteAddrs);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public ClusterAclVersionInfo examineBrokerClusterAclVersionInfo(
|
@Override
|
||||||
|
public ClusterAclVersionInfo examineBrokerClusterAclVersionInfo(
|
||||||
String addr) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
String addr) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public AclConfig examineBrokerClusterAclConfig(
|
@Override
|
||||||
|
public AclConfig examineBrokerClusterAclConfig(
|
||||||
String addr) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
String addr) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
||||||
return null;
|
return MQAdminInstance.threadLocalMQAdminExt().examineBrokerClusterAclConfig(addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -235,6 +240,12 @@ public class MQAdminExtImpl implements MQAdminExt {
|
|||||||
return MQAdminInstance.threadLocalMQAdminExt().wipeWritePermOfBroker(namesrvAddr, brokerName);
|
return MQAdminInstance.threadLocalMQAdminExt().wipeWritePermOfBroker(namesrvAddr, brokerName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int addWritePermOfBroker(String namesrvAddr,
|
||||||
|
String brokerName) throws RemotingCommandException, RemotingConnectException, RemotingSendRequestException, RemotingTimeoutException, InterruptedException, MQClientException {
|
||||||
|
return MQAdminInstance.threadLocalMQAdminExt().addWritePermOfBroker(namesrvAddr, brokerName);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void putKVConfig(String namespace, String key, String value) {
|
public void putKVConfig(String namespace, String key, String value) {
|
||||||
MQAdminInstance.threadLocalMQAdminExt().putKVConfig(namespace, key, value);
|
MQAdminInstance.threadLocalMQAdminExt().putKVConfig(namespace, key, value);
|
||||||
@@ -272,8 +283,9 @@ public class MQAdminExtImpl implements MQAdminExt {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void deleteSubscriptionGroup(String addr, String groupName, boolean removeOffset) throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
public void deleteSubscriptionGroup(String addr, String groupName, boolean removeOffset)
|
||||||
throw new UnsupportedOperationException();
|
throws RemotingException, MQBrokerException, InterruptedException, MQClientException {
|
||||||
|
MQAdminInstance.threadLocalMQAdminExt().deleteSubscriptionGroup(addr, groupName, removeOffset);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -511,9 +523,21 @@ public class MQAdminExtImpl implements MQAdminExt {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TopicConfigSerializeWrapper getAllTopicGroup(String brokerAddr,
|
public SubscriptionGroupWrapper getUserSubscriptionGroup(String brokerAddr,
|
||||||
long timeoutMillis) throws InterruptedException, RemotingTimeoutException, RemotingSendRequestException, RemotingConnectException, MQBrokerException {
|
long timeoutMillis) throws InterruptedException, RemotingTimeoutException, RemotingSendRequestException, RemotingConnectException, MQBrokerException {
|
||||||
return MQAdminInstance.threadLocalMQAdminExt().getAllTopicGroup(brokerAddr, timeoutMillis);
|
return MQAdminInstance.threadLocalMQAdminExt().getUserSubscriptionGroup(brokerAddr, timeoutMillis);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TopicConfigSerializeWrapper getAllTopicConfig(String brokerAddr,
|
||||||
|
long timeoutMillis) throws InterruptedException, RemotingTimeoutException, RemotingSendRequestException, RemotingConnectException, MQBrokerException {
|
||||||
|
return MQAdminInstance.threadLocalMQAdminExt().getAllTopicConfig(brokerAddr, timeoutMillis);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TopicConfigSerializeWrapper getUserTopicConfig(String brokerAddr, boolean specialTopic,
|
||||||
|
long timeoutMillis) throws InterruptedException, RemotingException, MQBrokerException, MQClientException {
|
||||||
|
return MQAdminInstance.threadLocalMQAdminExt().getUserTopicConfig(brokerAddr, specialTopic, timeoutMillis);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -0,0 +1,359 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.service.impl;
|
||||||
|
|
||||||
|
import com.google.common.base.Throwables;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.apache.commons.collections.CollectionUtils;
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
|
import org.apache.rocketmq.client.exception.MQBrokerException;
|
||||||
|
import org.apache.rocketmq.client.exception.MQClientException;
|
||||||
|
import org.apache.rocketmq.common.AclConfig;
|
||||||
|
import org.apache.rocketmq.common.MixAll;
|
||||||
|
import org.apache.rocketmq.common.PlainAccessConfig;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.ClusterInfo;
|
||||||
|
import org.apache.rocketmq.common.protocol.route.BrokerData;
|
||||||
|
import org.apache.rocketmq.dashboard.model.request.AclRequest;
|
||||||
|
import org.apache.rocketmq.dashboard.service.AbstractCommonService;
|
||||||
|
import org.apache.rocketmq.dashboard.service.AclService;
|
||||||
|
import org.apache.rocketmq.remoting.exception.RemotingConnectException;
|
||||||
|
import org.apache.rocketmq.remoting.exception.RemotingException;
|
||||||
|
import org.apache.rocketmq.remoting.exception.RemotingSendRequestException;
|
||||||
|
import org.apache.rocketmq.remoting.exception.RemotingTimeoutException;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
@Service
|
||||||
|
@Slf4j
|
||||||
|
public class AclServiceImpl extends AbstractCommonService implements AclService {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public AclConfig getAclConfig(boolean excludeSecretKey) {
|
||||||
|
try {
|
||||||
|
Optional<String> addr = getMasterSet().stream().findFirst();
|
||||||
|
if (addr.isPresent()) {
|
||||||
|
if (!excludeSecretKey) {
|
||||||
|
return mqAdminExt.examineBrokerClusterAclConfig(addr.get());
|
||||||
|
} else {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr.get());
|
||||||
|
if (CollectionUtils.isNotEmpty(aclConfig.getPlainAccessConfigs())) {
|
||||||
|
aclConfig.getPlainAccessConfigs().forEach(pac -> pac.setSecretKey(null));
|
||||||
|
}
|
||||||
|
return aclConfig;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("getAclConfig error.", e);
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
AclConfig aclConfig = new AclConfig();
|
||||||
|
aclConfig.setGlobalWhiteAddrs(Collections.emptyList());
|
||||||
|
aclConfig.setPlainAccessConfigs(Collections.emptyList());
|
||||||
|
return aclConfig;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void addAclConfig(PlainAccessConfig config) {
|
||||||
|
try {
|
||||||
|
Set<String> masterSet = getMasterSet();
|
||||||
|
|
||||||
|
if (masterSet.isEmpty()) {
|
||||||
|
throw new IllegalStateException("broker addr list is empty");
|
||||||
|
}
|
||||||
|
// check to see if account is exists
|
||||||
|
for (String addr : masterSet) {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr);
|
||||||
|
List<PlainAccessConfig> plainAccessConfigs = aclConfig.getPlainAccessConfigs();
|
||||||
|
for (PlainAccessConfig pac : plainAccessConfigs) {
|
||||||
|
if (pac.getAccessKey().equals(config.getAccessKey())) {
|
||||||
|
throw new IllegalArgumentException(String.format("broker: %s, exist accessKey: %s", addr, config.getAccessKey()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// all broker
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, config);
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void deleteAclConfig(PlainAccessConfig config) {
|
||||||
|
try {
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
log.info("Start to delete acl [{}] from broker [{}]", config.getAccessKey(), addr);
|
||||||
|
if (isExistAccessKey(config.getAccessKey(), addr)) {
|
||||||
|
mqAdminExt.deletePlainAccessConfig(addr, config.getAccessKey());
|
||||||
|
}
|
||||||
|
log.info("Delete acl [{}] from broker [{}] complete", config.getAccessKey(), addr);
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void updateAclConfig(PlainAccessConfig config) {
|
||||||
|
try {
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr);
|
||||||
|
if (aclConfig.getPlainAccessConfigs() != null) {
|
||||||
|
PlainAccessConfig remoteConfig = null;
|
||||||
|
for (PlainAccessConfig pac : aclConfig.getPlainAccessConfigs()) {
|
||||||
|
if (pac.getAccessKey().equals(config.getAccessKey())) {
|
||||||
|
remoteConfig = pac;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (remoteConfig != null) {
|
||||||
|
remoteConfig.setSecretKey(config.getSecretKey());
|
||||||
|
remoteConfig.setAdmin(config.isAdmin());
|
||||||
|
config = remoteConfig;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, config);
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void addOrUpdateAclTopicConfig(AclRequest request) {
|
||||||
|
try {
|
||||||
|
PlainAccessConfig addConfig = request.getConfig();
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr);
|
||||||
|
PlainAccessConfig remoteConfig = null;
|
||||||
|
if (aclConfig.getPlainAccessConfigs() != null) {
|
||||||
|
for (PlainAccessConfig config : aclConfig.getPlainAccessConfigs()) {
|
||||||
|
if (config.getAccessKey().equals(addConfig.getAccessKey())) {
|
||||||
|
remoteConfig = config;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (remoteConfig == null) {
|
||||||
|
// Maybe the broker no acl config of the access key, therefore add it;
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, addConfig);
|
||||||
|
} else {
|
||||||
|
if (remoteConfig.getTopicPerms() == null) {
|
||||||
|
remoteConfig.setTopicPerms(new ArrayList<>());
|
||||||
|
}
|
||||||
|
removeExist(remoteConfig.getTopicPerms(), request.getTopicPerm().split("=")[0]);
|
||||||
|
remoteConfig.getTopicPerms().add(request.getTopicPerm());
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, remoteConfig);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void addOrUpdateAclGroupConfig(AclRequest request) {
|
||||||
|
try {
|
||||||
|
PlainAccessConfig addConfig = request.getConfig();
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr);
|
||||||
|
PlainAccessConfig remoteConfig = null;
|
||||||
|
if (aclConfig.getPlainAccessConfigs() != null) {
|
||||||
|
for (PlainAccessConfig config : aclConfig.getPlainAccessConfigs()) {
|
||||||
|
if (config.getAccessKey().equals(addConfig.getAccessKey())) {
|
||||||
|
remoteConfig = config;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (remoteConfig == null) {
|
||||||
|
// May be the broker no acl config of the access key, therefore add it;
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, addConfig);
|
||||||
|
} else {
|
||||||
|
if (remoteConfig.getGroupPerms() == null) {
|
||||||
|
remoteConfig.setGroupPerms(new ArrayList<>());
|
||||||
|
}
|
||||||
|
removeExist(remoteConfig.getGroupPerms(), request.getGroupPerm().split("=")[0]);
|
||||||
|
remoteConfig.getGroupPerms().add(request.getGroupPerm());
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, remoteConfig);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void deletePermConfig(AclRequest request) {
|
||||||
|
try {
|
||||||
|
PlainAccessConfig deleteConfig = request.getConfig();
|
||||||
|
|
||||||
|
String topic = StringUtils.isNotEmpty(request.getTopicPerm()) ? request.getTopicPerm().split("=")[0] : null;
|
||||||
|
String group = StringUtils.isNotEmpty(request.getGroupPerm()) ? request.getGroupPerm().split("=")[0] : null;
|
||||||
|
if (deleteConfig.getTopicPerms() != null && topic != null) {
|
||||||
|
removeExist(deleteConfig.getTopicPerms(), topic);
|
||||||
|
}
|
||||||
|
if (deleteConfig.getGroupPerms() != null && group != null) {
|
||||||
|
removeExist(deleteConfig.getGroupPerms(), group);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr);
|
||||||
|
PlainAccessConfig remoteConfig = null;
|
||||||
|
if (aclConfig.getPlainAccessConfigs() != null) {
|
||||||
|
for (PlainAccessConfig config : aclConfig.getPlainAccessConfigs()) {
|
||||||
|
if (config.getAccessKey().equals(deleteConfig.getAccessKey())) {
|
||||||
|
remoteConfig = config;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (remoteConfig == null) {
|
||||||
|
// Maybe the broker no acl config of the access key, therefore add it;
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, deleteConfig);
|
||||||
|
} else {
|
||||||
|
if (remoteConfig.getTopicPerms() != null && topic != null) {
|
||||||
|
removeExist(remoteConfig.getTopicPerms(), topic);
|
||||||
|
}
|
||||||
|
if (remoteConfig.getGroupPerms() != null && group != null) {
|
||||||
|
removeExist(remoteConfig.getGroupPerms(), group);
|
||||||
|
}
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, remoteConfig);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void syncData(PlainAccessConfig config) {
|
||||||
|
try {
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
mqAdminExt.createAndUpdatePlainAccessConfig(addr, config);
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void addWhiteList(List<String> whiteList) {
|
||||||
|
if (whiteList == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr);
|
||||||
|
if (aclConfig.getGlobalWhiteAddrs() != null) {
|
||||||
|
aclConfig.setGlobalWhiteAddrs(Stream.of(whiteList, aclConfig.getGlobalWhiteAddrs()).flatMap(Collection::stream).distinct().collect(Collectors.toList()));
|
||||||
|
} else {
|
||||||
|
aclConfig.setGlobalWhiteAddrs(whiteList);
|
||||||
|
}
|
||||||
|
mqAdminExt.updateGlobalWhiteAddrConfig(addr, StringUtils.join(aclConfig.getGlobalWhiteAddrs(), ","));
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void deleteWhiteAddr(String deleteAddr) {
|
||||||
|
try {
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr);
|
||||||
|
if (aclConfig.getGlobalWhiteAddrs() == null || aclConfig.getGlobalWhiteAddrs().isEmpty()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
aclConfig.getGlobalWhiteAddrs().remove(deleteAddr);
|
||||||
|
mqAdminExt.updateGlobalWhiteAddrConfig(addr, StringUtils.join(aclConfig.getGlobalWhiteAddrs(), ","));
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void synchronizeWhiteList(List<String> whiteList) {
|
||||||
|
if (whiteList == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
for (String addr : getBrokerAddrs()) {
|
||||||
|
mqAdminExt.updateGlobalWhiteAddrConfig(addr, StringUtils.join(whiteList, ","));
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void removeExist(List<String> list, String name) {
|
||||||
|
Iterator<String> iterator = list.iterator();
|
||||||
|
while (iterator.hasNext()) {
|
||||||
|
String v = iterator.next();
|
||||||
|
String cmp = v.split("=")[0];
|
||||||
|
if (cmp.equals(name)) {
|
||||||
|
iterator.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isExistAccessKey(String accessKey,
|
||||||
|
String addr) throws InterruptedException, RemotingException, MQClientException, MQBrokerException {
|
||||||
|
AclConfig aclConfig = mqAdminExt.examineBrokerClusterAclConfig(addr);
|
||||||
|
List<PlainAccessConfig> plainAccessConfigs = aclConfig.getPlainAccessConfigs();
|
||||||
|
if (plainAccessConfigs == null || plainAccessConfigs.isEmpty()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (PlainAccessConfig config : plainAccessConfigs) {
|
||||||
|
if (accessKey.equals(config.getAccessKey())) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Set<BrokerData> getBrokerDataSet() throws InterruptedException, RemotingConnectException, RemotingTimeoutException, RemotingSendRequestException, MQBrokerException {
|
||||||
|
ClusterInfo clusterInfo = mqAdminExt.examineBrokerClusterInfo();
|
||||||
|
Map<String, BrokerData> brokerDataMap = clusterInfo.getBrokerAddrTable();
|
||||||
|
return new HashSet<>(brokerDataMap.values());
|
||||||
|
}
|
||||||
|
|
||||||
|
private Set<String> getMasterSet() throws InterruptedException, MQBrokerException, RemotingTimeoutException, RemotingSendRequestException, RemotingConnectException {
|
||||||
|
return getBrokerDataSet().stream().map(data -> data.getBrokerAddrs().get(MixAll.MASTER_ID)).collect(Collectors.toSet());
|
||||||
|
}
|
||||||
|
|
||||||
|
private Set<String> getBrokerAddrs() throws InterruptedException, MQBrokerException, RemotingTimeoutException, RemotingSendRequestException, RemotingConnectException {
|
||||||
|
Set<String> brokerAddrs = new HashSet<>();
|
||||||
|
getBrokerDataSet().forEach(data -> brokerAddrs.addAll(data.getBrokerAddrs().values()));
|
||||||
|
return brokerAddrs;
|
||||||
|
}
|
||||||
|
}
|
@@ -23,12 +23,14 @@ import com.google.common.collect.Iterables;
|
|||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
import com.google.common.collect.Maps;
|
import com.google.common.collect.Maps;
|
||||||
import com.google.common.collect.Sets;
|
import com.google.common.collect.Sets;
|
||||||
|
import java.util.Arrays;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
import javax.annotation.Resource;
|
||||||
import org.apache.commons.collections.CollectionUtils;
|
import org.apache.commons.collections.CollectionUtils;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.apache.rocketmq.client.exception.MQClientException;
|
import org.apache.rocketmq.client.exception.MQClientException;
|
||||||
@@ -46,6 +48,7 @@ import org.apache.rocketmq.common.protocol.body.GroupList;
|
|||||||
import org.apache.rocketmq.common.protocol.body.SubscriptionGroupWrapper;
|
import org.apache.rocketmq.common.protocol.body.SubscriptionGroupWrapper;
|
||||||
import org.apache.rocketmq.common.protocol.route.BrokerData;
|
import org.apache.rocketmq.common.protocol.route.BrokerData;
|
||||||
import org.apache.rocketmq.common.subscription.SubscriptionGroupConfig;
|
import org.apache.rocketmq.common.subscription.SubscriptionGroupConfig;
|
||||||
|
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
||||||
import org.apache.rocketmq.dashboard.model.ConsumerGroupRollBackStat;
|
import org.apache.rocketmq.dashboard.model.ConsumerGroupRollBackStat;
|
||||||
import org.apache.rocketmq.dashboard.model.GroupConsumeInfo;
|
import org.apache.rocketmq.dashboard.model.GroupConsumeInfo;
|
||||||
import org.apache.rocketmq.dashboard.model.QueueStatInfo;
|
import org.apache.rocketmq.dashboard.model.QueueStatInfo;
|
||||||
@@ -65,6 +68,9 @@ import static com.google.common.base.Throwables.propagate;
|
|||||||
public class ConsumerServiceImpl extends AbstractCommonService implements ConsumerService {
|
public class ConsumerServiceImpl extends AbstractCommonService implements ConsumerService {
|
||||||
private Logger logger = LoggerFactory.getLogger(ConsumerServiceImpl.class);
|
private Logger logger = LoggerFactory.getLogger(ConsumerServiceImpl.class);
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
private RMQConfigure configure;
|
||||||
|
|
||||||
private static final Set<String> SYSTEM_GROUP_SET = new HashSet<>();
|
private static final Set<String> SYSTEM_GROUP_SET = new HashSet<>();
|
||||||
|
|
||||||
static {
|
static {
|
||||||
@@ -290,11 +296,21 @@ public class ConsumerServiceImpl extends AbstractCommonService implements Consum
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean deleteSubGroup(DeleteSubGroupRequest deleteSubGroupRequest) {
|
public boolean deleteSubGroup(DeleteSubGroupRequest deleteSubGroupRequest) {
|
||||||
|
Set<String> brokerSet = this.fetchBrokerNameSetBySubscriptionGroup(deleteSubGroupRequest.getGroupName());
|
||||||
|
List<String> brokerList = deleteSubGroupRequest.getBrokerNameList();
|
||||||
|
boolean deleteInNsFlag = false;
|
||||||
|
// If the list of brokers passed in by the request contains the list of brokers that the consumer is in, delete RETRY and DLQ topic in namesrv
|
||||||
|
if (brokerList.containsAll(brokerSet)) {
|
||||||
|
deleteInNsFlag = true;
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
ClusterInfo clusterInfo = mqAdminExt.examineBrokerClusterInfo();
|
ClusterInfo clusterInfo = mqAdminExt.examineBrokerClusterInfo();
|
||||||
for (String brokerName : deleteSubGroupRequest.getBrokerNameList()) {
|
for (String brokerName : deleteSubGroupRequest.getBrokerNameList()) {
|
||||||
logger.info("addr={} groupName={}", clusterInfo.getBrokerAddrTable().get(brokerName).selectBrokerAddr(), deleteSubGroupRequest.getGroupName());
|
logger.info("addr={} groupName={}", clusterInfo.getBrokerAddrTable().get(brokerName).selectBrokerAddr(), deleteSubGroupRequest.getGroupName());
|
||||||
mqAdminExt.deleteSubscriptionGroup(clusterInfo.getBrokerAddrTable().get(brokerName).selectBrokerAddr(), deleteSubGroupRequest.getGroupName());
|
mqAdminExt.deleteSubscriptionGroup(clusterInfo.getBrokerAddrTable().get(brokerName).selectBrokerAddr(), deleteSubGroupRequest.getGroupName(), true);
|
||||||
|
// Delete %RETRY%+Group and %DLQ%+Group in broker and namesrv
|
||||||
|
deleteResources(MixAll.RETRY_GROUP_TOPIC_PREFIX + deleteSubGroupRequest.getGroupName(), brokerName, clusterInfo, deleteInNsFlag);
|
||||||
|
deleteResources(MixAll.DLQ_GROUP_TOPIC_PREFIX + deleteSubGroupRequest.getGroupName(), brokerName, clusterInfo, deleteInNsFlag);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception e) {
|
catch (Exception e) {
|
||||||
@@ -303,6 +319,18 @@ public class ConsumerServiceImpl extends AbstractCommonService implements Consum
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void deleteResources(String topic, String brokerName, ClusterInfo clusterInfo, boolean deleteInNsFlag) throws Exception {
|
||||||
|
mqAdminExt.deleteTopicInBroker(Sets.newHashSet(clusterInfo.getBrokerAddrTable().get(brokerName).selectBrokerAddr()), topic);
|
||||||
|
Set<String> nameServerSet = null;
|
||||||
|
if (StringUtils.isNotBlank(configure.getNamesrvAddr())) {
|
||||||
|
String[] ns = configure.getNamesrvAddr().split(";");
|
||||||
|
nameServerSet = new HashSet<>(Arrays.asList(ns));
|
||||||
|
}
|
||||||
|
if (deleteInNsFlag) {
|
||||||
|
mqAdminExt.deleteTopicInNameServer(nameServerSet, topic);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean createAndUpdateSubscriptionGroupConfig(ConsumerConfigInfo consumerConfigInfo) {
|
public boolean createAndUpdateSubscriptionGroupConfig(ConsumerConfigInfo consumerConfigInfo) {
|
||||||
try {
|
try {
|
||||||
|
@@ -19,12 +19,16 @@ package org.apache.rocketmq.dashboard.service.impl;
|
|||||||
|
|
||||||
import com.google.common.base.Throwables;
|
import com.google.common.base.Throwables;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.LinkedList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import org.apache.rocketmq.client.exception.MQClientException;
|
import org.apache.rocketmq.client.exception.MQClientException;
|
||||||
import org.apache.rocketmq.common.MixAll;
|
import org.apache.rocketmq.common.MixAll;
|
||||||
import org.apache.rocketmq.common.protocol.ResponseCode;
|
import org.apache.rocketmq.common.protocol.ResponseCode;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.ConsumeMessageDirectlyResult;
|
||||||
|
import org.apache.rocketmq.dashboard.model.DlqMessageResendResult;
|
||||||
|
import org.apache.rocketmq.dashboard.model.DlqMessageRequest;
|
||||||
import org.apache.rocketmq.dashboard.model.MessagePage;
|
import org.apache.rocketmq.dashboard.model.MessagePage;
|
||||||
import org.apache.rocketmq.dashboard.model.MessageView;
|
import org.apache.rocketmq.dashboard.model.MessageView;
|
||||||
import org.apache.rocketmq.dashboard.model.request.MessageQuery;
|
import org.apache.rocketmq.dashboard.model.request.MessageQuery;
|
||||||
@@ -65,4 +69,17 @@ public class DlqMessageServiceImpl implements DlqMessageService {
|
|||||||
}
|
}
|
||||||
return messageService.queryMessageByPage(query);
|
return messageService.queryMessageByPage(query);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<DlqMessageResendResult> batchResendDlqMessage(List<DlqMessageRequest> dlqMessages) {
|
||||||
|
List<DlqMessageResendResult> batchResendResults = new LinkedList<>();
|
||||||
|
for (DlqMessageRequest dlqMessage : dlqMessages) {
|
||||||
|
ConsumeMessageDirectlyResult result = messageService.consumeMessageDirectly(dlqMessage.getTopicName(),
|
||||||
|
dlqMessage.getMsgId(), dlqMessage.getConsumerGroup(),
|
||||||
|
dlqMessage.getClientId());
|
||||||
|
DlqMessageResendResult resendResult = new DlqMessageResendResult(result, dlqMessage.getMsgId());
|
||||||
|
batchResendResults.add(resendResult);
|
||||||
|
}
|
||||||
|
return batchResendResults;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -31,6 +31,7 @@ import org.apache.rocketmq.acl.common.SessionCredentials;
|
|||||||
import org.apache.rocketmq.client.consumer.DefaultMQPullConsumer;
|
import org.apache.rocketmq.client.consumer.DefaultMQPullConsumer;
|
||||||
import org.apache.rocketmq.client.consumer.PullResult;
|
import org.apache.rocketmq.client.consumer.PullResult;
|
||||||
import org.apache.rocketmq.client.consumer.PullStatus;
|
import org.apache.rocketmq.client.consumer.PullStatus;
|
||||||
|
import org.apache.rocketmq.client.exception.MQClientException;
|
||||||
import org.apache.rocketmq.common.MixAll;
|
import org.apache.rocketmq.common.MixAll;
|
||||||
import org.apache.rocketmq.common.Pair;
|
import org.apache.rocketmq.common.Pair;
|
||||||
import org.apache.rocketmq.common.message.MessageClientIDSetter;
|
import org.apache.rocketmq.common.message.MessageClientIDSetter;
|
||||||
@@ -111,6 +112,9 @@ public class MessageServiceImpl implements MessageService {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
} catch (Exception err) {
|
} catch (Exception err) {
|
||||||
|
if (err instanceof MQClientException) {
|
||||||
|
throw new ServiceException(-1, ((MQClientException) err).getErrorMessage());
|
||||||
|
}
|
||||||
throw Throwables.propagate(err);
|
throw Throwables.propagate(err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -325,7 +329,7 @@ public class MessageServiceImpl implements MessageService {
|
|||||||
List<MessageExt> msgFoundList = pullResult.getMsgFoundList();
|
List<MessageExt> msgFoundList = pullResult.getMsgFoundList();
|
||||||
for (int i = msgFoundList.size() - 1; i >= 0; i--) {
|
for (int i = msgFoundList.size() - 1; i >= 0; i--) {
|
||||||
MessageExt messageExt = msgFoundList.get(i);
|
MessageExt messageExt = msgFoundList.get(i);
|
||||||
if (messageExt.getStoreTimestamp() < query.getBegin()) {
|
if (messageExt.getStoreTimestamp() > query.getEnd()) {
|
||||||
end--;
|
end--;
|
||||||
} else {
|
} else {
|
||||||
hasIllegalOffset = false;
|
hasIllegalOffset = false;
|
||||||
|
@@ -20,8 +20,6 @@ package org.apache.rocketmq.dashboard.service.impl;
|
|||||||
import com.google.common.base.Function;
|
import com.google.common.base.Function;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
|
|
||||||
import com.google.common.base.Throwables;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
@@ -35,6 +33,7 @@ import org.apache.rocketmq.common.Pair;
|
|||||||
import org.apache.rocketmq.common.message.MessageExt;
|
import org.apache.rocketmq.common.message.MessageExt;
|
||||||
import org.apache.rocketmq.common.topic.TopicValidator;
|
import org.apache.rocketmq.common.topic.TopicValidator;
|
||||||
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
||||||
|
import org.apache.rocketmq.dashboard.exception.ServiceException;
|
||||||
import org.apache.rocketmq.dashboard.model.MessageTraceView;
|
import org.apache.rocketmq.dashboard.model.MessageTraceView;
|
||||||
import org.apache.rocketmq.dashboard.model.trace.ProducerNode;
|
import org.apache.rocketmq.dashboard.model.trace.ProducerNode;
|
||||||
import org.apache.rocketmq.dashboard.model.trace.MessageTraceGraph;
|
import org.apache.rocketmq.dashboard.model.trace.MessageTraceGraph;
|
||||||
@@ -82,7 +81,7 @@ public class MessageTraceServiceImpl implements MessageTraceService {
|
|||||||
}
|
}
|
||||||
return messageTraceViews;
|
return messageTraceViews;
|
||||||
} catch (Exception err) {
|
} catch (Exception err) {
|
||||||
throw Throwables.propagate(err);
|
throw new ServiceException(-1, String.format("Failed to query message trace by msgId %s", key));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
package org.apache.rocketmq.dashboard.service.impl;
|
package org.apache.rocketmq.dashboard.service.impl;
|
||||||
|
|
||||||
import com.google.common.base.Splitter;
|
|
||||||
import com.google.common.collect.Maps;
|
import com.google.common.collect.Maps;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
@@ -46,7 +45,8 @@ public class OpsServiceImpl extends AbstractCommonService implements OpsService
|
|||||||
@Override
|
@Override
|
||||||
public Map<String, Object> homePageInfo() {
|
public Map<String, Object> homePageInfo() {
|
||||||
Map<String, Object> homePageInfoMap = Maps.newHashMap();
|
Map<String, Object> homePageInfoMap = Maps.newHashMap();
|
||||||
homePageInfoMap.put("namesvrAddrList", Splitter.on(";").splitToList(configure.getNamesrvAddr()));
|
homePageInfoMap.put("currentNamesrv", configure.getNamesrvAddr());
|
||||||
|
homePageInfoMap.put("namesvrAddrList", configure.getNamesrvAddrs());
|
||||||
homePageInfoMap.put("useVIPChannel", Boolean.valueOf(configure.getIsVIPChannel()));
|
homePageInfoMap.put("useVIPChannel", Boolean.valueOf(configure.getIsVIPChannel()));
|
||||||
homePageInfoMap.put("useTLS", configure.isUseTLS());
|
homePageInfoMap.put("useTLS", configure.isUseTLS());
|
||||||
return homePageInfoMap;
|
return homePageInfoMap;
|
||||||
@@ -73,7 +73,8 @@ public class OpsServiceImpl extends AbstractCommonService implements OpsService
|
|||||||
return checkResultMap;
|
return checkResultMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public boolean updateIsVIPChannel(String useVIPChannel) {
|
@Override
|
||||||
|
public boolean updateIsVIPChannel(String useVIPChannel) {
|
||||||
configure.setIsVIPChannel(useVIPChannel);
|
configure.setIsVIPChannel(useVIPChannel);
|
||||||
mqAdminExtPool.clear();
|
mqAdminExtPool.clear();
|
||||||
return true;
|
return true;
|
||||||
@@ -85,4 +86,13 @@ public class OpsServiceImpl extends AbstractCommonService implements OpsService
|
|||||||
mqAdminExtPool.clear();
|
mqAdminExtPool.clear();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void addNameSvrAddr(String namesrvAddr) {
|
||||||
|
List<String> namesrvAddrs = configure.getNamesrvAddrs();
|
||||||
|
if (namesrvAddrs != null && !namesrvAddrs.contains(namesrvAddr)) {
|
||||||
|
namesrvAddrs.add(namesrvAddr);
|
||||||
|
}
|
||||||
|
configure.setNamesrvAddrs(namesrvAddrs);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -135,10 +135,10 @@ public class TopicServiceImpl extends AbstractCommonService implements TopicServ
|
|||||||
ClusterInfo clusterInfo = null;
|
ClusterInfo clusterInfo = null;
|
||||||
try {
|
try {
|
||||||
clusterInfo = mqAdminExt.examineBrokerClusterInfo();
|
clusterInfo = mqAdminExt.examineBrokerClusterInfo();
|
||||||
|
return mqAdminExt.examineTopicConfig(clusterInfo.getBrokerAddrTable().get(brokerName).selectBrokerAddr(), topic);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw Throwables.propagate(e);
|
throw Throwables.propagate(e);
|
||||||
}
|
}
|
||||||
return mqAdminExt.examineTopicConfig(clusterInfo.getBrokerAddrTable().get(brokerName).selectBrokerAddr(), topic);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -0,0 +1,108 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.task;
|
||||||
|
|
||||||
|
import com.google.common.base.Throwables;
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.ExecutionException;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.apache.rocketmq.common.MixAll;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.BrokerStatsData;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.GroupList;
|
||||||
|
import org.apache.rocketmq.common.protocol.route.BrokerData;
|
||||||
|
import org.apache.rocketmq.common.protocol.route.TopicRouteData;
|
||||||
|
import org.apache.rocketmq.dashboard.service.DashboardCollectService;
|
||||||
|
import org.apache.rocketmq.store.stats.BrokerStatsManager;
|
||||||
|
import org.apache.rocketmq.tools.admin.MQAdminExt;
|
||||||
|
import org.apache.rocketmq.tools.command.stats.StatsAllSubCommand;
|
||||||
|
|
||||||
|
@Slf4j
|
||||||
|
public class CollectTaskRunnble implements Runnable {
|
||||||
|
|
||||||
|
private String topic;
|
||||||
|
|
||||||
|
private MQAdminExt mqAdminExt;
|
||||||
|
|
||||||
|
private DashboardCollectService dashboardCollectService;
|
||||||
|
|
||||||
|
public CollectTaskRunnble(String topic, MQAdminExt mqAdminExt,
|
||||||
|
DashboardCollectService dashboardCollectService) {
|
||||||
|
this.topic = topic;
|
||||||
|
this.mqAdminExt = mqAdminExt;
|
||||||
|
this.dashboardCollectService = dashboardCollectService;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
Date date = new Date();
|
||||||
|
try {
|
||||||
|
TopicRouteData topicRouteData = mqAdminExt.examineTopicRouteInfo(topic);
|
||||||
|
GroupList groupList = mqAdminExt.queryTopicConsumeByWho(topic);
|
||||||
|
double inTPS = 0;
|
||||||
|
long inMsgCntToday = 0;
|
||||||
|
double outTPS = 0;
|
||||||
|
long outMsgCntToday = 0;
|
||||||
|
for (BrokerData bd : topicRouteData.getBrokerDatas()) {
|
||||||
|
String masterAddr = bd.getBrokerAddrs().get(MixAll.MASTER_ID);
|
||||||
|
if (masterAddr != null) {
|
||||||
|
try {
|
||||||
|
BrokerStatsData bsd = mqAdminExt.viewBrokerStatsData(masterAddr, BrokerStatsManager.TOPIC_PUT_NUMS, topic);
|
||||||
|
inTPS += bsd.getStatsMinute().getTps();
|
||||||
|
inMsgCntToday += StatsAllSubCommand.compute24HourSum(bsd);
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.warn("Exception caught: mqAdminExt get broker stats data TOPIC_PUT_NUMS failed, topic [{}]", topic, e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (groupList != null && !groupList.getGroupList().isEmpty()) {
|
||||||
|
for (String group : groupList.getGroupList()) {
|
||||||
|
for (BrokerData bd : topicRouteData.getBrokerDatas()) {
|
||||||
|
String masterAddr = bd.getBrokerAddrs().get(MixAll.MASTER_ID);
|
||||||
|
if (masterAddr != null) {
|
||||||
|
try {
|
||||||
|
String statsKey = String.format("%s@%s", topic, group);
|
||||||
|
BrokerStatsData bsd = mqAdminExt.viewBrokerStatsData(masterAddr, BrokerStatsManager.GROUP_GET_NUMS, statsKey);
|
||||||
|
outTPS += bsd.getStatsMinute().getTps();
|
||||||
|
outMsgCntToday += StatsAllSubCommand.compute24HourSum(bsd);
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.warn("Exception caught: mqAdminExt get broker stats data GROUP_GET_NUMS failed, topic [{}], group [{}]", topic, group, e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
List<String> list;
|
||||||
|
try {
|
||||||
|
list = dashboardCollectService.getTopicMap().get(topic);
|
||||||
|
} catch (ExecutionException e) {
|
||||||
|
throw Throwables.propagate(e);
|
||||||
|
}
|
||||||
|
if (null == list) {
|
||||||
|
list = Lists.newArrayList();
|
||||||
|
}
|
||||||
|
|
||||||
|
list.add(date.getTime() + "," + new BigDecimal(inTPS).setScale(5, BigDecimal.ROUND_HALF_UP) + "," + inMsgCntToday + "," + new BigDecimal(outTPS).setScale(5, BigDecimal.ROUND_HALF_UP) + "," + outMsgCntToday);
|
||||||
|
dashboardCollectService.getTopicMap().put(topic, list);
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("Failed to collect topic: {} data", topic, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -16,17 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
package org.apache.rocketmq.dashboard.task;
|
package org.apache.rocketmq.dashboard.task;
|
||||||
|
|
||||||
import com.google.common.base.Stopwatch;
|
|
||||||
import org.apache.rocketmq.common.protocol.body.ClusterInfo;
|
|
||||||
import org.apache.rocketmq.common.protocol.body.GroupList;
|
|
||||||
import org.apache.rocketmq.common.protocol.body.KVTable;
|
|
||||||
import org.apache.rocketmq.common.protocol.body.TopicList;
|
|
||||||
import org.apache.rocketmq.common.protocol.route.BrokerData;
|
|
||||||
import org.apache.rocketmq.common.protocol.route.TopicRouteData;
|
|
||||||
import org.apache.rocketmq.common.topic.TopicValidator;
|
|
||||||
import org.apache.rocketmq.store.stats.BrokerStatsManager;
|
|
||||||
import org.apache.rocketmq.tools.admin.MQAdminExt;
|
|
||||||
import org.apache.rocketmq.tools.command.stats.StatsAllSubCommand;
|
|
||||||
import com.google.common.base.Throwables;
|
import com.google.common.base.Throwables;
|
||||||
import com.google.common.cache.LoadingCache;
|
import com.google.common.cache.LoadingCache;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
@@ -42,13 +31,18 @@ import java.util.HashMap;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.ExecutionException;
|
import java.util.concurrent.ExecutorService;
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
import org.apache.rocketmq.common.MixAll;
|
import org.apache.rocketmq.common.MixAll;
|
||||||
import org.apache.rocketmq.common.protocol.body.BrokerStatsData;
|
import org.apache.rocketmq.common.protocol.body.ClusterInfo;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.KVTable;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.TopicList;
|
||||||
|
import org.apache.rocketmq.common.protocol.route.BrokerData;
|
||||||
|
import org.apache.rocketmq.common.topic.TopicValidator;
|
||||||
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
||||||
import org.apache.rocketmq.dashboard.service.DashboardCollectService;
|
import org.apache.rocketmq.dashboard.service.DashboardCollectService;
|
||||||
import org.apache.rocketmq.dashboard.util.JsonUtil;
|
import org.apache.rocketmq.dashboard.util.JsonUtil;
|
||||||
|
import org.apache.rocketmq.tools.admin.MQAdminExt;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.scheduling.annotation.Scheduled;
|
import org.springframework.scheduling.annotation.Scheduled;
|
||||||
@@ -67,14 +61,14 @@ public class DashboardCollectTask {
|
|||||||
|
|
||||||
private final static Logger log = LoggerFactory.getLogger(DashboardCollectTask.class);
|
private final static Logger log = LoggerFactory.getLogger(DashboardCollectTask.class);
|
||||||
|
|
||||||
|
@Resource
|
||||||
|
private ExecutorService collectExecutor;
|
||||||
|
|
||||||
@Scheduled(cron = "30 0/1 * * * ?")
|
@Scheduled(cron = "30 0/1 * * * ?")
|
||||||
public void collectTopic() {
|
public void collectTopic() {
|
||||||
if (!rmqConfigure.isEnableDashBoardCollect()) {
|
if (!rmqConfigure.isEnableDashBoardCollect()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Date date = new Date();
|
|
||||||
Stopwatch stopwatch = Stopwatch.createUnstarted();
|
|
||||||
try {
|
try {
|
||||||
TopicList topicList = mqAdminExt.fetchAllTopicList();
|
TopicList topicList = mqAdminExt.fetchAllTopicList();
|
||||||
Set<String> topicSet = topicList.getTopicList();
|
Set<String> topicSet = topicList.getTopicList();
|
||||||
@@ -85,77 +79,9 @@ public class DashboardCollectTask {
|
|||||||
|| TopicValidator.isSystemTopic(topic)) {
|
|| TopicValidator.isSystemTopic(topic)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
TopicRouteData topicRouteData = mqAdminExt.examineTopicRouteInfo(topic);
|
CollectTaskRunnble collectTask = new CollectTaskRunnble(topic, mqAdminExt, dashboardCollectService);
|
||||||
|
collectExecutor.submit(collectTask);
|
||||||
GroupList groupList = mqAdminExt.queryTopicConsumeByWho(topic);
|
|
||||||
|
|
||||||
double inTPS = 0;
|
|
||||||
|
|
||||||
long inMsgCntToday = 0;
|
|
||||||
|
|
||||||
double outTPS = 0;
|
|
||||||
|
|
||||||
long outMsgCntToday = 0;
|
|
||||||
|
|
||||||
for (BrokerData bd : topicRouteData.getBrokerDatas()) {
|
|
||||||
String masterAddr = bd.getBrokerAddrs().get(MixAll.MASTER_ID);
|
|
||||||
if (masterAddr != null) {
|
|
||||||
try {
|
|
||||||
stopwatch.start();
|
|
||||||
log.info("start time: {}", stopwatch.toString());
|
|
||||||
BrokerStatsData bsd = mqAdminExt.viewBrokerStatsData(masterAddr, BrokerStatsManager.TOPIC_PUT_NUMS, topic);
|
|
||||||
stopwatch.stop();
|
|
||||||
log.info("stop time : {}", stopwatch.toString());
|
|
||||||
|
|
||||||
inTPS += bsd.getStatsMinute().getTps();
|
|
||||||
inMsgCntToday += StatsAllSubCommand.compute24HourSum(bsd);
|
|
||||||
}
|
|
||||||
catch (Exception e) {
|
|
||||||
stopwatch.reset();
|
|
||||||
log.warn("Exception caught: mqAdminExt get broker stats data TOPIC_PUT_NUMS failed");
|
|
||||||
log.warn("Response [{}] ", e.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (groupList != null && !groupList.getGroupList().isEmpty()) {
|
|
||||||
|
|
||||||
for (String group : groupList.getGroupList()) {
|
|
||||||
for (BrokerData bd : topicRouteData.getBrokerDatas()) {
|
|
||||||
String masterAddr = bd.getBrokerAddrs().get(MixAll.MASTER_ID);
|
|
||||||
if (masterAddr != null) {
|
|
||||||
try {
|
|
||||||
String statsKey = String.format("%s@%s", topic, group);
|
|
||||||
BrokerStatsData bsd = mqAdminExt.viewBrokerStatsData(masterAddr, BrokerStatsManager.GROUP_GET_NUMS, statsKey);
|
|
||||||
outTPS += bsd.getStatsMinute().getTps();
|
|
||||||
outMsgCntToday += StatsAllSubCommand.compute24HourSum(bsd);
|
|
||||||
}
|
|
||||||
catch (Exception e) {
|
|
||||||
log.warn("Exception caught: mqAdminExt get broker stats data GROUP_GET_NUMS failed");
|
|
||||||
log.warn("Response [{}] ", e.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
List<String> list;
|
|
||||||
try {
|
|
||||||
list = dashboardCollectService.getTopicMap().get(topic);
|
|
||||||
}
|
|
||||||
catch (ExecutionException e) {
|
|
||||||
throw Throwables.propagate(e);
|
|
||||||
}
|
|
||||||
if (null == list) {
|
|
||||||
list = Lists.newArrayList();
|
|
||||||
}
|
|
||||||
|
|
||||||
list.add(date.getTime() + "," + new BigDecimal(inTPS).setScale(5, BigDecimal.ROUND_HALF_UP) + "," + inMsgCntToday + "," + new BigDecimal(outTPS).setScale(5, BigDecimal.ROUND_HALF_UP) + "," + outMsgCntToday);
|
|
||||||
dashboardCollectService.getTopicMap().put(topic, list);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
log.debug("Topic Collected Data in memory = {}" + JsonUtil.obj2String(dashboardCollectService.getTopicMap().asMap()));
|
|
||||||
}
|
}
|
||||||
catch (Exception err) {
|
catch (Exception err) {
|
||||||
throw Throwables.propagate(err);
|
throw Throwables.propagate(err);
|
||||||
@@ -328,6 +254,7 @@ public class DashboardCollectTask {
|
|||||||
for (Map.Entry<String, Set<String>> entry : clusterTable.entrySet()) {
|
for (Map.Entry<String, Set<String>> entry : clusterTable.entrySet()) {
|
||||||
String clusterName = entry.getKey();
|
String clusterName = entry.getKey();
|
||||||
TopicValidator.addSystemTopic(clusterName);
|
TopicValidator.addSystemTopic(clusterName);
|
||||||
|
TopicValidator.addSystemTopic(MixAll.getReplyTopic(clusterName));
|
||||||
Set<String> brokerNames = entry.getValue();
|
Set<String> brokerNames = entry.getValue();
|
||||||
for (String brokerName : brokerNames) {
|
for (String brokerName : brokerNames) {
|
||||||
TopicValidator.addSystemTopic(brokerName);
|
TopicValidator.addSystemTopic(brokerName);
|
||||||
|
@@ -1,54 +0,0 @@
|
|||||||
#
|
|
||||||
# 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.
|
|
||||||
#
|
|
||||||
|
|
||||||
server.address=0.0.0.0
|
|
||||||
server.port=8080
|
|
||||||
|
|
||||||
### SSL setting
|
|
||||||
#server.ssl.key-store=classpath:rmqcngkeystore.jks
|
|
||||||
#server.ssl.key-store-password=rocketmq
|
|
||||||
#server.ssl.keyStoreType=PKCS12
|
|
||||||
#server.ssl.keyAlias=rmqcngkey
|
|
||||||
|
|
||||||
#spring.application.index=true
|
|
||||||
spring.application.name=rocketmq-dashboard
|
|
||||||
spring.http.encoding.charset=UTF-8
|
|
||||||
spring.http.encoding.enabled=true
|
|
||||||
spring.http.encoding.force=true
|
|
||||||
logging.level.root=INFO
|
|
||||||
logging.config=classpath:logback.xml
|
|
||||||
#if this value is empty,use env value rocketmq.config.namesrvAddr NAMESRV_ADDR | now, you can set it in ops page.default localhost:9876
|
|
||||||
rocketmq.config.namesrvAddr=
|
|
||||||
#if you use rocketmq version < 3.5.8, rocketmq.config.isVIPChannel should be false.default true
|
|
||||||
rocketmq.config.isVIPChannel=
|
|
||||||
#timeout for mqadminExt, default 5000ms
|
|
||||||
rocketmq.config.timeoutMillis=
|
|
||||||
#rocketmq-console's data path:dashboard/monitor
|
|
||||||
rocketmq.config.dataPath=/tmp/rocketmq-console/data
|
|
||||||
#set it false if you don't want use dashboard.default true
|
|
||||||
rocketmq.config.enableDashBoardCollect=true
|
|
||||||
#set the message track trace topic if you don't want use the default one
|
|
||||||
rocketmq.config.msgTrackTopicName=
|
|
||||||
rocketmq.config.ticketKey=ticket
|
|
||||||
|
|
||||||
#Must create userInfo file: ${rocketmq.config.dataPath}/users.properties if the login is required
|
|
||||||
rocketmq.config.loginRequired=false
|
|
||||||
|
|
||||||
#set the accessKey and secretKey if you used acl
|
|
||||||
#rocketmq.config.accessKey=
|
|
||||||
#rocketmq.config.secretKey=
|
|
||||||
rocketmq.config.useTLS=false
|
|
69
src/main/resources/application.yml
Normal file
69
src/main/resources/application.yml
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
||||||
|
|
||||||
|
server:
|
||||||
|
port: 8080
|
||||||
|
servlet:
|
||||||
|
encoding:
|
||||||
|
charset: UTF-8
|
||||||
|
enabled: true
|
||||||
|
force: true
|
||||||
|
## SSL setting
|
||||||
|
# ssl:
|
||||||
|
# key-store: classpath:rmqcngkeystore.jks
|
||||||
|
# key-store-password: rocketmq
|
||||||
|
# key-store-type: PKCS12
|
||||||
|
# key-alias: rmqcngkey
|
||||||
|
|
||||||
|
spring:
|
||||||
|
application:
|
||||||
|
name: rocketmq-dashboard
|
||||||
|
|
||||||
|
logging:
|
||||||
|
config: classpath:logback.xml
|
||||||
|
|
||||||
|
rocketmq:
|
||||||
|
config:
|
||||||
|
# if this value is empty,use env value rocketmq.config.namesrvAddr NAMESRV_ADDR | now, default localhost:9876
|
||||||
|
# configure multiple namesrv addresses to manage multiple different clusters
|
||||||
|
namesrvAddrs:
|
||||||
|
- 127.0.0.1:9876
|
||||||
|
- 127.0.0.2:9876
|
||||||
|
# if you use rocketmq version < 3.5.8, rocketmq.config.isVIPChannel should be false.default true
|
||||||
|
isVIPChannel:
|
||||||
|
# timeout for mqadminExt, default 5000ms
|
||||||
|
timeoutMillis:
|
||||||
|
# rocketmq-console's data path:dashboard/monitor
|
||||||
|
dataPath: /tmp/rocketmq-console/data
|
||||||
|
# set it false if you don't want use dashboard.default true
|
||||||
|
enableDashBoardCollect: true
|
||||||
|
# set the message track trace topic if you don't want use the default one
|
||||||
|
msgTrackTopicName:
|
||||||
|
ticketKey: ticket
|
||||||
|
# must create userInfo file: ${rocketmq.config.dataPath}/users.properties if the login is required
|
||||||
|
loginRequired: false
|
||||||
|
useTLS: false
|
||||||
|
# set the accessKey and secretKey if you used acl
|
||||||
|
accessKey: # if version > 4.4.0
|
||||||
|
secretKey: # if version > 4.4.0
|
||||||
|
|
||||||
|
threadpool:
|
||||||
|
config:
|
||||||
|
coreSize: 10
|
||||||
|
maxSize: 10
|
||||||
|
keepAliveTime: 3000
|
||||||
|
queueSize: 5000
|
@@ -19,7 +19,7 @@
|
|||||||
<configuration>
|
<configuration>
|
||||||
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
|
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
|
||||||
<encoder charset="UTF-8">
|
<encoder charset="UTF-8">
|
||||||
<pattern>[%d{yyyy-MM-dd HH:mm:ss.SSS}] %5p %m%n</pattern>
|
<pattern>[%d{yyyy-MM-dd HH:mm:ss.SSS}] %p %t - %m%n</pattern>
|
||||||
</encoder>
|
</encoder>
|
||||||
</appender>
|
</appender>
|
||||||
|
|
||||||
@@ -37,7 +37,7 @@
|
|||||||
<MaxHistory>10</MaxHistory>
|
<MaxHistory>10</MaxHistory>
|
||||||
</rollingPolicy>
|
</rollingPolicy>
|
||||||
<encoder>
|
<encoder>
|
||||||
<pattern>[%d{yyyy-MM-dd HH:mm:ss.SSS}] %5p %m%n</pattern>
|
<pattern>[%d{yyyy-MM-dd HH:mm:ss.SSS}] %p %t - %m%n</pattern>
|
||||||
<charset class="java.nio.charset.Charset">UTF-8</charset>
|
<charset class="java.nio.charset.Charset">UTF-8</charset>
|
||||||
</encoder>
|
</encoder>
|
||||||
</appender>
|
</appender>
|
||||||
|
@@ -23,13 +23,18 @@
|
|||||||
|
|
||||||
rolePerms:
|
rolePerms:
|
||||||
ordinary:
|
ordinary:
|
||||||
- /rocketmq/nsaddr
|
- /rocketmq/*.query
|
||||||
- /ops/*
|
- /ops/*.query
|
||||||
- /dashboard/**
|
- /dashboard/*.query
|
||||||
- /topic/*.query
|
- /topic/*.query
|
||||||
- /topic/sendTopicMessage.do
|
- /topic/sendTopicMessage.do
|
||||||
- /producer/*.query
|
- /producer/*.query
|
||||||
- /message/*
|
- /message/*.query
|
||||||
- /messageTrace/*
|
- /messageTrace/*.query
|
||||||
- /monitor/*
|
- /monitor/*.query
|
||||||
|
- /consumer/*.query
|
||||||
|
- /cluster/*.query
|
||||||
|
- /dlqMessage/*.query
|
||||||
|
- /dlqMessage/exportDlqMessage.do
|
||||||
|
- /dlqMessage/batchResendDlqMessage.do
|
||||||
|
- /acl/*.query
|
||||||
|
@@ -113,5 +113,6 @@
|
|||||||
<script type="text/javascript" src="src/remoteApi/remoteApi.js"></script>
|
<script type="text/javascript" src="src/remoteApi/remoteApi.js"></script>
|
||||||
<script type="text/javascript" src="vendor/preLoading/main.js"></script>
|
<script type="text/javascript" src="vendor/preLoading/main.js"></script>
|
||||||
<script type="text/javascript" src="src/login.js"></script>
|
<script type="text/javascript" src="src/login.js"></script>
|
||||||
|
<script type="text/javascript" src="src/acl.js"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
540
src/main/resources/static/src/acl.js
Normal file
540
src/main/resources/static/src/acl.js
Normal file
@@ -0,0 +1,540 @@
|
|||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
var module = app;
|
||||||
|
|
||||||
|
module.controller('aclController', ['$scope', 'ngDialog', '$http', 'Notification', '$window', function ($scope, ngDialog, $http, Notification, $window) {
|
||||||
|
$scope.paginationConf = {
|
||||||
|
currentPage: 1,
|
||||||
|
totalItems: 0,
|
||||||
|
itemsPerPage: 10,
|
||||||
|
pagesLength: 15,
|
||||||
|
perPageOptions: [10],
|
||||||
|
rememberPerPage: 'perPageItems',
|
||||||
|
onChange: function () {
|
||||||
|
$scope.showPlainAccessConfigs(this.currentPage, this.totalItems);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.plainAccessConfigs = [];
|
||||||
|
$scope.allPlainAccessConfigs = [];
|
||||||
|
$scope.globalWhiteAddrs = [];
|
||||||
|
$scope.allGlobalWhiteAddrs = [];
|
||||||
|
$scope.userRole = $window.sessionStorage.getItem("userrole");
|
||||||
|
$scope.writeOperationEnabled = $scope.userRole == null ? true : ($scope.userRole == 1 ? true : false);
|
||||||
|
$scope.showSecretKeyType = {};
|
||||||
|
|
||||||
|
$scope.refreshPlainAccessConfigs = function () {
|
||||||
|
$http({
|
||||||
|
method: "GET",
|
||||||
|
url: "acl/config.query",
|
||||||
|
params: {}
|
||||||
|
}).success(function (resp) {
|
||||||
|
|
||||||
|
// globalWhiteAddrs
|
||||||
|
// plainAccessConfigs
|
||||||
|
if (resp.status == 0) {
|
||||||
|
$scope.allPlainAccessConfigs = resp.data.plainAccessConfigs;
|
||||||
|
$scope.allGlobalWhiteAddrs = resp.data.globalWhiteAddrs;
|
||||||
|
$scope.showSecretKeyType = {};
|
||||||
|
$scope.allPlainAccessConfigs.forEach(e => $scope.showSecretKeyType[e.accessKey] = {
|
||||||
|
type: 'password',
|
||||||
|
action: 'SHOW'
|
||||||
|
});
|
||||||
|
$scope.showPlainAccessConfigs(1, $scope.allPlainAccessConfigs.length);
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
$scope.filterStr = "";
|
||||||
|
$scope.$watch('filterStr', function () {
|
||||||
|
$scope.paginationConf.currentPage = 1;
|
||||||
|
$scope.filterList(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
$scope.filterList = function (currentPage) {
|
||||||
|
var lowExceptStr = $scope.filterStr.toLowerCase();
|
||||||
|
var canShowList = [];
|
||||||
|
|
||||||
|
$scope.allPlainAccessConfigs.forEach(function (element) {
|
||||||
|
if (element.accessKey.toLowerCase().indexOf(lowExceptStr) != -1) {
|
||||||
|
canShowList.push(element);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$scope.paginationConf.totalItems = canShowList.length;
|
||||||
|
var perPage = $scope.paginationConf.itemsPerPage;
|
||||||
|
var from = (currentPage - 1) * perPage;
|
||||||
|
var to = (from + perPage) > canShowList.length ? canShowList.length : from + perPage;
|
||||||
|
$scope.plainAccessConfigs = canShowList.slice(from, to);
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.showPlainAccessConfigs = function (currentPage, totalItem) {
|
||||||
|
var perPage = $scope.paginationConf.itemsPerPage;
|
||||||
|
var from = (currentPage - 1) * perPage;
|
||||||
|
var to = (from + perPage) > totalItem ? totalItem : from + perPage;
|
||||||
|
$scope.plainAccessConfigs = $scope.allPlainAccessConfigs.slice(from, to);
|
||||||
|
$scope.paginationConf.totalItems = totalItem;
|
||||||
|
$scope.filterList($scope.paginationConf.currentPage)
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// add acl account
|
||||||
|
$scope.openAddDialog = function () {
|
||||||
|
var request = {};
|
||||||
|
request.accessKey = '';
|
||||||
|
request.secretKey = '';
|
||||||
|
request.admin = false;
|
||||||
|
request.defaultTopicPerm = 'DENY';
|
||||||
|
request.defaultGroupPerm = 'SUB';
|
||||||
|
ngDialog.open({
|
||||||
|
preCloseCallback: function (value) {
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
},
|
||||||
|
template: 'addAclAccountDialog',
|
||||||
|
controller: 'addAclAccountDialogController',
|
||||||
|
data: request
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.deleteAclConfig = function (accessKey) {
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/delete.do",
|
||||||
|
data: {accessKey: accessKey}
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
$scope.openUpdateDialog = function (request) {
|
||||||
|
ngDialog.open({
|
||||||
|
preCloseCallback: function (value) {
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
},
|
||||||
|
template: 'updateAclAccountDialog',
|
||||||
|
controller: 'updateAclAccountDialogController',
|
||||||
|
data: request
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// add acl topic permission
|
||||||
|
$scope.openAddTopicDialog = function (request) {
|
||||||
|
$.extend(request, {pub: true, sub: true, deny: false})
|
||||||
|
ngDialog.open({
|
||||||
|
preCloseCallback: function (value) {
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
},
|
||||||
|
template: 'addAclTopicDialog',
|
||||||
|
controller: 'addAclTopicDialogController',
|
||||||
|
data: request
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// update acl topic permission
|
||||||
|
$scope.openUpdateTopicDialog = function (request, topic) {
|
||||||
|
var perm = {pub: false, sub: false, deny: false};
|
||||||
|
var topicInfo = topic.split('=');
|
||||||
|
$.each(topicInfo[1].split('|'), function (i, e) {
|
||||||
|
switch (e) {
|
||||||
|
case 'PUB':
|
||||||
|
perm.pub = true;
|
||||||
|
break;
|
||||||
|
case 'SUB':
|
||||||
|
perm.sub = true;
|
||||||
|
break;
|
||||||
|
case 'DENY':
|
||||||
|
perm.deny = true;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$.extend(request, perm, {topic: topicInfo[0]});
|
||||||
|
ngDialog.open({
|
||||||
|
preCloseCallback: function (value) {
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
},
|
||||||
|
template: 'updateAclTopicDialog',
|
||||||
|
controller: 'updateAclTopicDialogController',
|
||||||
|
data: request
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// add acl group permission
|
||||||
|
$scope.openAddGroupDialog = function (request) {
|
||||||
|
$.extend(request, {pub: true, sub: true, deny: false})
|
||||||
|
ngDialog.open({
|
||||||
|
preCloseCallback: function (value) {
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
},
|
||||||
|
template: 'addAclGroupDialog',
|
||||||
|
controller: 'addAclGroupDialogController',
|
||||||
|
data: request
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// update acl group permission
|
||||||
|
$scope.openUpdateGroupDialog = function (request, group) {
|
||||||
|
var perm = {pub: false, sub: false, deny: false};
|
||||||
|
var groupInfo = group.split('=');
|
||||||
|
$.each(groupInfo[1].split('|'), function (i, e) {
|
||||||
|
switch (e) {
|
||||||
|
case 'PUB':
|
||||||
|
perm.pub = true;
|
||||||
|
break;
|
||||||
|
case 'SUB':
|
||||||
|
perm.sub = true;
|
||||||
|
break;
|
||||||
|
case 'DENY':
|
||||||
|
perm.deny = true;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$.extend(request, perm, {group: groupInfo[0]});
|
||||||
|
ngDialog.open({
|
||||||
|
preCloseCallback: function (value) {
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
},
|
||||||
|
template: 'updateAclGroupDialog',
|
||||||
|
controller: 'updateAclGroupDialogController',
|
||||||
|
data: request
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.deletePermConfig = function (config, name, type) {
|
||||||
|
var request = {config: config};
|
||||||
|
switch (type) {
|
||||||
|
case 'topic':
|
||||||
|
request.topicPerm = name;
|
||||||
|
break;
|
||||||
|
case 'group':
|
||||||
|
request.groupPerm = name;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/perm/delete.do",
|
||||||
|
data: request
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.synchronizeData = function (request) {
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/sync.do",
|
||||||
|
data: request
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.openAddAddrDialog = function () {
|
||||||
|
ngDialog.open({
|
||||||
|
preCloseCallback: function (value) {
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
},
|
||||||
|
template: 'addWhiteListDialog',
|
||||||
|
controller: 'addWhiteListDialogController'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.deleteGlobalWhiteAddr = function (request) {
|
||||||
|
$http({
|
||||||
|
method: "DELETE",
|
||||||
|
url: "acl/white/list/delete.do?request=" + request
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.synchronizeWhiteList = function (request) {
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/white/list/sync.do",
|
||||||
|
data: request
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
$scope.refreshPlainAccessConfigs();
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.switchSecretKeyType = function (accessKey) {
|
||||||
|
if ($scope.showSecretKeyType[accessKey].type == 'password') {
|
||||||
|
$scope.showSecretKeyType[accessKey] = {type: 'text', action: 'HIDE'};
|
||||||
|
} else {
|
||||||
|
$scope.showSecretKeyType[accessKey] = {type: 'password', action: 'SHOW'};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}]);
|
||||||
|
|
||||||
|
module.controller('addAclAccountDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.addRequest = function (requestItem) {
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/add.do",
|
||||||
|
data: requestItem
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
);
|
||||||
|
|
||||||
|
module.controller('updateAclAccountDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.updateAclAccountRequest = function (requestItem) {
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/update.do",
|
||||||
|
data: requestItem
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
);
|
||||||
|
|
||||||
|
module.controller('addAclTopicDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.updateAclAccountRequest = function (requestItem) {
|
||||||
|
if ((requestItem.deny && requestItem.sub) || (requestItem.deny && requestItem.pub)) {
|
||||||
|
alert("Forbid deny && pub/sub.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!requestItem.topic) {
|
||||||
|
alert("topic is null");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
//var request = requestItem.originalData;
|
||||||
|
var originalData = $.extend(true, {}, requestItem.originalData);
|
||||||
|
if (!originalData.topicPerms) {
|
||||||
|
originalData.topicPerms = new Array();
|
||||||
|
}
|
||||||
|
var topicPerm = concatPerm(requestItem.topic, requestItem.pub ? 0x01 : 0, requestItem.sub ? 0x02 : 0, requestItem.deny ? 0x04 : 0);
|
||||||
|
originalData.topicPerms.push(topicPerm);
|
||||||
|
var request = {topicPerm: topicPerm, config: originalData};
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/topic/add.do",
|
||||||
|
data: request
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
);
|
||||||
|
|
||||||
|
module.controller('updateAclTopicDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.updateAclAccountRequest = function (requestItem) {
|
||||||
|
if ((requestItem.deny && requestItem.sub) || (requestItem.deny && requestItem.pub)) {
|
||||||
|
alert("Forbid deny && pub/sub.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
var originalData = $.extend(true, {}, requestItem.originalData);
|
||||||
|
if (!originalData.topicPerms) {
|
||||||
|
originalData.topicPerms = new Array();
|
||||||
|
}
|
||||||
|
var topicPerm = concatPerm(requestItem.topic, requestItem.pub ? 0x01 : 0, requestItem.sub ? 0x02 : 0, requestItem.deny ? 0x04 : 0);
|
||||||
|
|
||||||
|
for (var i = 0; i < originalData.topicPerms.length; i++) {
|
||||||
|
if (originalData.topicPerms[i].split('=')[0] == requestItem.topic) {
|
||||||
|
originalData.topicPerms[i] = topicPerm;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var request = {topicPerm: topicPerm, config: originalData};
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/topic/add.do",
|
||||||
|
data: request
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
);
|
||||||
|
|
||||||
|
module.controller('addAclGroupDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.updateAclAccountRequest = function (requestItem) {
|
||||||
|
if ((requestItem.deny && requestItem.sub) || (requestItem.deny && requestItem.pub)) {
|
||||||
|
alert("Forbid deny && pub/sub.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
//var request = requestItem.originalData;
|
||||||
|
var originalData = $.extend(true, {}, requestItem.originalData);
|
||||||
|
if (!originalData.groupPerms) {
|
||||||
|
originalData.groupPerms = new Array();
|
||||||
|
}
|
||||||
|
var groupPerm = concatPerm(requestItem.group, requestItem.pub ? 0x01 : 0, requestItem.sub ? 0x02 : 0, requestItem.deny ? 0x04 : 0);
|
||||||
|
originalData.groupPerms.push(groupPerm);
|
||||||
|
var request = {groupPerm: groupPerm, config: originalData};
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/group/add.do",
|
||||||
|
data: request
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
);
|
||||||
|
|
||||||
|
module.controller('updateAclGroupDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.updateAclAccountRequest = function (requestItem) {
|
||||||
|
if ((requestItem.deny && requestItem.sub) || (requestItem.deny && requestItem.pub)) {
|
||||||
|
alert("Forbid deny && pub/sub.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
var originalData = $.extend(true, {}, requestItem.originalData);
|
||||||
|
if (!originalData.groupPerms) {
|
||||||
|
originalData.groupPerms = new Array();
|
||||||
|
}
|
||||||
|
var groupPerm = concatPerm(requestItem.group, requestItem.pub ? 0x01 : 0, requestItem.sub ? 0x02 : 0, requestItem.deny ? 0x04 : 0);
|
||||||
|
|
||||||
|
for (var i = 0; i < originalData.groupPerms.length; i++) {
|
||||||
|
if (originalData.groupPerms[i].split('=')[0] == requestItem.group) {
|
||||||
|
originalData.groupPerms[i] = groupPerm;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var request = {groupPerm: groupPerm, config: originalData};
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/group/add.do",
|
||||||
|
data: request
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* pub: 0x01, sub: 0x02, deny: 0x04
|
||||||
|
*/
|
||||||
|
function concatPerm(name, pub, sub, deny) {
|
||||||
|
var perm = '';
|
||||||
|
|
||||||
|
switch (pub | sub | deny) {
|
||||||
|
case 0x01:
|
||||||
|
perm = 'PUB';
|
||||||
|
break;
|
||||||
|
case 0x02:
|
||||||
|
perm = 'SUB';
|
||||||
|
break;
|
||||||
|
case 0x03:
|
||||||
|
perm = 'PUB|SUB';
|
||||||
|
break;
|
||||||
|
case 0x04:
|
||||||
|
perm = 'DENY';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
perm = 'DENY';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return name + '=' + perm;
|
||||||
|
}
|
||||||
|
|
||||||
|
module.controller('addWhiteListDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.addWhiteListRequest = function (requestItem) {
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/white/list/add.do",
|
||||||
|
data: requestItem.split(',')
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
);
|
||||||
|
|
||||||
|
module.controller('aclBelongItemDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.postBelongItemRequest = function (topicRequestItem) {
|
||||||
|
topicRequestItem.type = 1
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "acl/belong/item/add.do",
|
||||||
|
data: topicRequestItem
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
Notification.info({message: "success!", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
);
|
@@ -49,6 +49,15 @@ var app = angular.module('app', [
|
|||||||
}
|
}
|
||||||
console.log('initFlag0='+ initFlag + ' loginFlag0==='+loginFlag);
|
console.log('initFlag0='+ initFlag + ' loginFlag0==='+loginFlag);
|
||||||
|
|
||||||
|
$http({
|
||||||
|
method: "GET",
|
||||||
|
url: "acl/enable.query"
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp && resp.status == 0) {
|
||||||
|
$rootScope.show = resp.data;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
$rootScope.$on('$locationChangeStart', function (event, next, current) {
|
$rootScope.$on('$locationChangeStart', function (event, next, current) {
|
||||||
// redirect to login page if not logged in and trying to access a restricted page
|
// redirect to login page if not logged in and trying to access a restricted page
|
||||||
init(function(resp){
|
init(function(resp){
|
||||||
@@ -204,6 +213,9 @@ app.config(['$routeProvider', '$httpProvider','$cookiesProvider','getDictNamePro
|
|||||||
}).when('/ops', {
|
}).when('/ops', {
|
||||||
templateUrl: 'view/pages/ops.html',
|
templateUrl: 'view/pages/ops.html',
|
||||||
controller:'opsController'
|
controller:'opsController'
|
||||||
|
}).when('/acl', {
|
||||||
|
templateUrl: 'view/pages/acl.html',
|
||||||
|
controller: 'aclController'
|
||||||
}).when('/404', {
|
}).when('/404', {
|
||||||
templateUrl: 'view/pages/404.html'
|
templateUrl: 'view/pages/404.html'
|
||||||
}).otherwise('/404');
|
}).otherwise('/404');
|
||||||
|
@@ -82,6 +82,9 @@ module.controller('dlqMessageController', ['$scope', 'ngDialog', '$http', 'Notif
|
|||||||
if ($scope.messageShowList.length == 0){
|
if ($scope.messageShowList.length == 0){
|
||||||
$("#noMsgTip").removeAttr("style");
|
$("#noMsgTip").removeAttr("style");
|
||||||
}
|
}
|
||||||
|
for (const message of $scope.messageShowList) {
|
||||||
|
message.checked = false;
|
||||||
|
}
|
||||||
console.log($scope.messageShowList);
|
console.log($scope.messageShowList);
|
||||||
if (resp.data.page.first) {
|
if (resp.data.page.first) {
|
||||||
$scope.paginationConf.currentPage = 1;
|
$scope.paginationConf.currentPage = 1;
|
||||||
@@ -180,5 +183,115 @@ module.controller('dlqMessageController', ['$scope', 'ngDialog', '$http', 'Notif
|
|||||||
|
|
||||||
$scope.exportDlqMessage = function (msgId, consumerGroup) {
|
$scope.exportDlqMessage = function (msgId, consumerGroup) {
|
||||||
window.location.href = "dlqMessage/exportDlqMessage.do?msgId=" + msgId + "&consumerGroup=" + consumerGroup;
|
window.location.href = "dlqMessage/exportDlqMessage.do?msgId=" + msgId + "&consumerGroup=" + consumerGroup;
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.selectedDlqMessage = [];
|
||||||
|
$scope.batchResendDlqMessage = function (consumerGroup) {
|
||||||
|
if ($("#batchResendBtn").hasClass("disabled")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (const message of $scope.messageCheckedList) {
|
||||||
|
const dlqMessage = {};
|
||||||
|
dlqMessage.topic = message.properties.RETRY_TOPIC;
|
||||||
|
dlqMessage.msgId = message.properties.ORIGIN_MESSAGE_ID;
|
||||||
|
dlqMessage.consumerGroup = consumerGroup;
|
||||||
|
$scope.selectedDlqMessage.push(dlqMessage);
|
||||||
|
}
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "dlqMessage/batchResendDlqMessage.do",
|
||||||
|
data: $scope.selectedDlqMessage
|
||||||
|
}).success(function (resp) {
|
||||||
|
$scope.selectedDlqMessage = [];
|
||||||
|
if (resp.status == 0) {
|
||||||
|
ngDialog.open({
|
||||||
|
template: 'operationResultDialog',
|
||||||
|
data: {
|
||||||
|
result: resp.data
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
ngDialog.open({
|
||||||
|
template: 'operationResultDialog',
|
||||||
|
data: {
|
||||||
|
result: resp.errMsg
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.batchExportDlqMessage = function (consumerGroup) {
|
||||||
|
if ($("#batchExportBtn").hasClass("disabled")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (const message of $scope.messageCheckedList) {
|
||||||
|
const dlqMessage = {};
|
||||||
|
dlqMessage.msgId = message.msgId;
|
||||||
|
dlqMessage.consumerGroup = consumerGroup;
|
||||||
|
$scope.selectedDlqMessage.push(dlqMessage);
|
||||||
|
}
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "dlqMessage/batchExportDlqMessage.do",
|
||||||
|
data: $scope.selectedDlqMessage,
|
||||||
|
headers: {
|
||||||
|
'Content-type': 'application/json'
|
||||||
|
},
|
||||||
|
responseType: "arraybuffer"
|
||||||
|
}).success(function (resp) {
|
||||||
|
$scope.selectedDlqMessage = [];
|
||||||
|
const blob = new Blob([resp], {type: "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"});
|
||||||
|
const objectUrl = URL.createObjectURL(blob);
|
||||||
|
const a = document.createElement('a');
|
||||||
|
a.style.display = 'none';
|
||||||
|
a.download = 'dlqs.xlsx';
|
||||||
|
a.href = objectUrl;
|
||||||
|
a.click();
|
||||||
|
document.body.removeChild(a)
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
$scope.checkedAll = false;
|
||||||
|
$scope.messageCheckedList = [];
|
||||||
|
$scope.selectAll = function () {
|
||||||
|
$scope.messageCheckedList = [];
|
||||||
|
if ($scope.checkedAll == true) {
|
||||||
|
angular.forEach($scope.messageShowList, function (item, index) {
|
||||||
|
item.checked = true;
|
||||||
|
$scope.messageCheckedList.push(item);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
angular.forEach($scope.messageShowList, function (item, index) {
|
||||||
|
item.checked = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
checkBtn($scope.messageCheckedList)
|
||||||
|
console.log($scope.messageCheckedList)
|
||||||
|
}
|
||||||
|
|
||||||
|
$scope.selectItem = function () {
|
||||||
|
var flag = true;
|
||||||
|
$scope.messageCheckedList = [];
|
||||||
|
angular.forEach($scope.messageShowList, function (item, index) {
|
||||||
|
if (item.checked) {
|
||||||
|
$scope.messageCheckedList.push(item);
|
||||||
|
} else {
|
||||||
|
flag = false;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
$scope.checkedAll = flag;
|
||||||
|
checkBtn($scope.messageCheckedList)
|
||||||
|
console.log($scope.messageCheckedList);
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkBtn(messageCheckList) {
|
||||||
|
if (messageCheckList.length == 0) {
|
||||||
|
$("#batchResendBtn").addClass("disabled");
|
||||||
|
$("#batchExportBtn").addClass("disabled");
|
||||||
|
} else {
|
||||||
|
$("#batchResendBtn").removeClass("disabled");
|
||||||
|
$("#batchExportBtn").removeClass("disabled");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}]);
|
}]);
|
@@ -21,8 +21,8 @@ var en = {
|
|||||||
"NO": "NO.",
|
"NO": "NO.",
|
||||||
"ADDRESS": "Address",
|
"ADDRESS": "Address",
|
||||||
"VERSION": "Version",
|
"VERSION": "Version",
|
||||||
"PRO_MSG_TPS": "Produce Massage TPS",
|
"PRO_MSG_TPS": "Produce Message TPS",
|
||||||
"CUS_MSG_TPS": "Consumer Massage TPS",
|
"CUS_MSG_TPS": "Consumer Message TPS",
|
||||||
"YESTERDAY_PRO_COUNT": "Yesterday Produce Count",
|
"YESTERDAY_PRO_COUNT": "Yesterday Produce Count",
|
||||||
"YESTERDAY_CUS_COUNT": "Yesterday Consume Count",
|
"YESTERDAY_CUS_COUNT": "Yesterday Consume Count",
|
||||||
"TODAY_PRO_COUNT": "Today Produce Count",
|
"TODAY_PRO_COUNT": "Today Produce Count",
|
||||||
@@ -111,5 +111,17 @@ var en = {
|
|||||||
"TRACE_TOPIC":"TraceTopic",
|
"TRACE_TOPIC":"TraceTopic",
|
||||||
"SELECT_TRACE_TOPIC":"selectTraceTopic",
|
"SELECT_TRACE_TOPIC":"selectTraceTopic",
|
||||||
"EXPORT": "export",
|
"EXPORT": "export",
|
||||||
"NO_MATCH_RESULT": "no match result"
|
"NO_MATCH_RESULT": "no match result",
|
||||||
}
|
"BATCH_RESEND": "batchReSend",
|
||||||
|
"BATCH_EXPORT": "batchExport",
|
||||||
|
"WHITE_LIST":"White List",
|
||||||
|
"ACCOUNT_INFO":"Account Info",
|
||||||
|
"IS_ADMIN":"Is Admin",
|
||||||
|
"DEFAULT_TOPIC_PERM":"Default Topic Permission",
|
||||||
|
"DEFAULT_GROUP_PERM":"Default Group Permission",
|
||||||
|
"TOPIC_PERM":"Topic Permission",
|
||||||
|
"GROUP_PERM":"Group Permission",
|
||||||
|
"SYNCHRONIZE":"Synchronize Data",
|
||||||
|
"SHOW":"Show",
|
||||||
|
"HIDE":"Hide"
|
||||||
|
}
|
||||||
|
@@ -112,5 +112,17 @@ var zh = {
|
|||||||
"TRACE_TOPIC":"消息轨迹主题",
|
"TRACE_TOPIC":"消息轨迹主题",
|
||||||
"SELECT_TRACE_TOPIC":"选择消息轨迹主题",
|
"SELECT_TRACE_TOPIC":"选择消息轨迹主题",
|
||||||
"EXPORT": "导出",
|
"EXPORT": "导出",
|
||||||
"NO_MATCH_RESULT": "没有查到符合条件的结果"
|
"NO_MATCH_RESULT": "没有查到符合条件的结果",
|
||||||
|
"BATCH_RESEND": "批量重发",
|
||||||
|
"BATCH_EXPORT": "批量导出",
|
||||||
|
"WHITE_LIST":"白名单",
|
||||||
|
"ACCOUNT_INFO":"账户信息",
|
||||||
|
"IS_ADMIN":"是否管理员",
|
||||||
|
"DEFAULT_TOPIC_PERM":"topic默认权限",
|
||||||
|
"DEFAULT_GROUP_PERM":"消费组默认权限",
|
||||||
|
"TOPIC_PERM":"topic权限",
|
||||||
|
"GROUP_PERM":"消费组权限",
|
||||||
|
"SYNCHRONIZE":"同步",
|
||||||
|
"SHOW":"显示",
|
||||||
|
"HIDE":"隐藏"
|
||||||
}
|
}
|
@@ -58,6 +58,7 @@ module.controller('messageController', ['$scope', 'ngDialog', '$http', 'Notifica
|
|||||||
};
|
};
|
||||||
|
|
||||||
$scope.queryMessagePageByTopic = function () {
|
$scope.queryMessagePageByTopic = function () {
|
||||||
|
$("#noMsgTip").css("display", "none");
|
||||||
if ($scope.timepickerEnd < $scope.timepickerBegin) {
|
if ($scope.timepickerEnd < $scope.timepickerBegin) {
|
||||||
Notification.error({message: "endTime is later than beginTime!", delay: 2000});
|
Notification.error({message: "endTime is later than beginTime!", delay: 2000});
|
||||||
return
|
return
|
||||||
@@ -80,6 +81,9 @@ module.controller('messageController', ['$scope', 'ngDialog', '$http', 'Notifica
|
|||||||
if (resp.status === 0) {
|
if (resp.status === 0) {
|
||||||
console.log(resp);
|
console.log(resp);
|
||||||
$scope.messageShowList = resp.data.page.content;
|
$scope.messageShowList = resp.data.page.content;
|
||||||
|
if ($scope.messageShowList.length == 0){
|
||||||
|
$("#noMsgTip").removeAttr("style");
|
||||||
|
}
|
||||||
if (resp.data.page.first) {
|
if (resp.data.page.first) {
|
||||||
$scope.paginationConf.currentPage = 1;
|
$scope.paginationConf.currentPage = 1;
|
||||||
}
|
}
|
||||||
@@ -206,7 +210,8 @@ module.controller('messageController', ['$scope', 'ngDialog', '$http', 'Notifica
|
|||||||
}]);
|
}]);
|
||||||
|
|
||||||
module.controller('messageDetailViewDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
module.controller('messageDetailViewDialogController', ['$scope', 'ngDialog', '$http', 'Notification', function ($scope, ngDialog, $http, Notification) {
|
||||||
|
$scope.messageTrackList = $scope.ngDialogData.messageTrackList;
|
||||||
|
$scope.messageTrackShowList = $scope.ngDialogData.messageTrackList;
|
||||||
$scope.resendMessage = function (messageView, consumerGroup) {
|
$scope.resendMessage = function (messageView, consumerGroup) {
|
||||||
var topic = messageView.topic;
|
var topic = messageView.topic;
|
||||||
var msgId = messageView.msgId;
|
var msgId = messageView.msgId;
|
||||||
@@ -258,5 +263,18 @@ module.controller('messageDetailViewDialogController', ['$scope', 'ngDialog', '$
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
$scope.filterConsumerGroup = "";
|
||||||
|
$scope.$watch('filterConsumerGroup', function () {
|
||||||
|
const lowExceptStr = $scope.filterConsumerGroup.toLowerCase();
|
||||||
|
const canShowList = [];
|
||||||
|
|
||||||
|
$scope.messageTrackList.forEach(function (element) {
|
||||||
|
if (element.consumerGroup.toLowerCase().indexOf(lowExceptStr) != -1) {
|
||||||
|
canShowList.push(element);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$scope.messageTrackShowList = canShowList;
|
||||||
|
});
|
||||||
}]
|
}]
|
||||||
);
|
);
|
@@ -22,6 +22,7 @@ app.controller('opsController', ['$scope', '$location', '$http', 'Notification',
|
|||||||
$scope.userRole = $window.sessionStorage.getItem("userrole");
|
$scope.userRole = $window.sessionStorage.getItem("userrole");
|
||||||
$scope.writeOperationEnabled = $scope.userRole == null ? true : ($scope.userRole == 1 ? true : false);
|
$scope.writeOperationEnabled = $scope.userRole == null ? true : ($scope.userRole == 1 ? true : false);
|
||||||
$scope.inputReadonly = !$scope.writeOperationEnabled;
|
$scope.inputReadonly = !$scope.writeOperationEnabled;
|
||||||
|
$scope.newNamesrvAddr = "";
|
||||||
$http({
|
$http({
|
||||||
method: "GET",
|
method: "GET",
|
||||||
url: "ops/homePage.query"
|
url: "ops/homePage.query"
|
||||||
@@ -30,6 +31,7 @@ app.controller('opsController', ['$scope', '$location', '$http', 'Notification',
|
|||||||
$scope.namesvrAddrList = resp.data.namesvrAddrList;
|
$scope.namesvrAddrList = resp.data.namesvrAddrList;
|
||||||
$scope.useVIPChannel = resp.data.useVIPChannel;
|
$scope.useVIPChannel = resp.data.useVIPChannel;
|
||||||
$scope.useTLS = resp.data.useTLS;
|
$scope.useTLS = resp.data.useTLS;
|
||||||
|
$scope.selectedNamesrv = resp.data.currentNamesrv;
|
||||||
} else {
|
} else {
|
||||||
Notification.error({message: resp.errMsg, delay: 2000});
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
}
|
}
|
||||||
@@ -43,7 +45,7 @@ app.controller('opsController', ['$scope', '$location', '$http', 'Notification',
|
|||||||
$http({
|
$http({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
url: "ops/updateNameSvrAddr.do",
|
url: "ops/updateNameSvrAddr.do",
|
||||||
params: {nameSvrAddrList: $scope.namesvrAddrList.join(";")}
|
params: {nameSvrAddrList: $scope.selectedNamesrv}
|
||||||
}).success(function (resp) {
|
}).success(function (resp) {
|
||||||
if (resp.status == 0) {
|
if (resp.status == 0) {
|
||||||
Notification.info({message: "SUCCESS", delay: 2000});
|
Notification.info({message: "SUCCESS", delay: 2000});
|
||||||
@@ -52,6 +54,26 @@ app.controller('opsController', ['$scope', '$location', '$http', 'Notification',
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
$scope.addNameSvrAddr = function () {
|
||||||
|
$http({
|
||||||
|
method: "POST",
|
||||||
|
url: "ops/addNameSvrAddr.do",
|
||||||
|
params: {newNamesrvAddr: $scope.newNamesrvAddr}
|
||||||
|
}).success(function (resp) {
|
||||||
|
if (resp.status == 0) {
|
||||||
|
if ($scope.namesvrAddrList.indexOf($scope.newNamesrvAddr) == -1) {
|
||||||
|
$scope.namesvrAddrList.push($scope.newNamesrvAddr);
|
||||||
|
}
|
||||||
|
$("#namesrvAddr").val("");
|
||||||
|
$scope.newNamesrvAddr = "";
|
||||||
|
Notification.info({message: "SUCCESS", delay: 2000});
|
||||||
|
} else {
|
||||||
|
Notification.error({message: resp.errMsg, delay: 2000});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
$scope.updateIsVIPChannel = function () {
|
$scope.updateIsVIPChannel = function () {
|
||||||
$http({
|
$http({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
|
@@ -54,7 +54,7 @@ app.service('remoteApi', ['$http','tools', function ($http,tools) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var queryTopicCurrentData = function(callback){
|
var queryTopicCurrentData = function(callback){
|
||||||
var url = 'dashboard/topicCurrent';
|
var url = 'dashboard/topicCurrent.query';
|
||||||
var setting = {
|
var setting = {
|
||||||
type: "GET",
|
type: "GET",
|
||||||
timeout:15000,//data is too large,so master set time out is long enough
|
timeout:15000,//data is too large,so master set time out is long enough
|
||||||
|
@@ -289,3 +289,15 @@
|
|||||||
.table.text-middle>tbody>tr>td,.table.text-middle>tbody>tr>th{
|
.table.text-middle>tbody>tr>td,.table.text-middle>tbody>tr>th{
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.perm-table{width: 100%;}
|
||||||
|
.perm-table .perm-tg{width: 70%;}
|
||||||
|
.perm-table .center{border-left: 1px solid #e4dddd; border-right: 1px solid #e4dddd;}
|
||||||
|
|
||||||
|
.input-none {
|
||||||
|
border: 0;
|
||||||
|
outline: none;
|
||||||
|
background-color: rgba(0, 0, 0, 0);
|
||||||
|
cursor: text !important;
|
||||||
|
width: 60%;
|
||||||
|
}
|
@@ -36,6 +36,7 @@
|
|||||||
<li ng-class="path =='message' ? 'active':''"><a ng-href="#/message">{{'MESSAGE' | translate}}</a></li>
|
<li ng-class="path =='message' ? 'active':''"><a ng-href="#/message">{{'MESSAGE' | translate}}</a></li>
|
||||||
<li ng-class="path =='dlqMessage' ? 'active':''"><a ng-href="#/dlqMessage">{{'DLQ_MESSAGE' | translate}}</a></li>
|
<li ng-class="path =='dlqMessage' ? 'active':''"><a ng-href="#/dlqMessage">{{'DLQ_MESSAGE' | translate}}</a></li>
|
||||||
<li ng-class="path =='messageTrace' ? 'active':''"><a ng-href="#/messageTrace">{{'MESSAGETRACE' | translate}}</a></li>
|
<li ng-class="path =='messageTrace' ? 'active':''"><a ng-href="#/messageTrace">{{'MESSAGETRACE' | translate}}</a></li>
|
||||||
|
<li ng-show="{{ show }}" ng-class="path =='acl' ? 'active':''"><a ng-href="#/acl">Acl</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
<ul class="nav navbar-nav navbar-right">
|
<ul class="nav navbar-nav navbar-right">
|
||||||
<li class="dropdown">
|
<li class="dropdown">
|
||||||
|
483
src/main/resources/static/view/pages/acl.html
Normal file
483
src/main/resources/static/view/pages/acl.html
Normal file
@@ -0,0 +1,483 @@
|
|||||||
|
<!--
|
||||||
|
~ 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.
|
||||||
|
-->
|
||||||
|
<div class="container-fluid" id="deployHistoryList">
|
||||||
|
<div class="modal-body">
|
||||||
|
<div ng-cloak="" class="tabsdemoDynamicHeight">
|
||||||
|
<md-content>
|
||||||
|
<md-tabs md-dynamic-height="" md-border-bottom="">
|
||||||
|
<md-tab label="Account Info">
|
||||||
|
<md-content class="md-padding" style="min-height:600px">
|
||||||
|
<form class="form-inline pull-left col-sm-12">
|
||||||
|
<div class="form-group">
|
||||||
|
<label>Access Key:</label>
|
||||||
|
<input class="form-control" style="width: 450px" type="text" ng-model="filterStr"/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button ng-show="writeOperationEnabled" class="btn btn-raised btn-sm btn-primary"
|
||||||
|
type="button"
|
||||||
|
ng-click="openAddDialog()">{{'ADD' |
|
||||||
|
translate}}
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
<table class="table table-bordered">
|
||||||
|
<tr>
|
||||||
|
<th class="text-center">Access Key</th>
|
||||||
|
<th ng-show="writeOperationEnabled" class="text-center">Secret Key</th>
|
||||||
|
<th class="text-center">{{'IS_ADMIN' | translate}}</th>
|
||||||
|
<th class="text-center">{{'DEFAULT_TOPIC_PERM' | translate}}</th>
|
||||||
|
<th class="text-center">{{'DEFAULT_GROUP_PERM' | translate}}</th>
|
||||||
|
<th class="text-center">{{'TOPIC_PERM' | translate}}</th>
|
||||||
|
<th class="text-center">{{'GROUP_PERM' | translate}}</th>
|
||||||
|
<th ng-show="writeOperationEnabled" class="text-center">
|
||||||
|
{{'OPERATION' | translate}}
|
||||||
|
</th>
|
||||||
|
</tr>
|
||||||
|
<tr ng-repeat="item in plainAccessConfigs">
|
||||||
|
<td class="text-center">{{item.accessKey}}</td>
|
||||||
|
<td ng-show="writeOperationEnabled" class="text-center">
|
||||||
|
<input type="{{showSecretKeyType[item.accessKey].type}}"
|
||||||
|
value="{{item.secretKey}}" class="input-none" ng-disabled="true"/>
|
||||||
|
<a href="javascript:;"
|
||||||
|
ng-click="switchSecretKeyType(item.accessKey)">{{showSecretKeyType[item.accessKey].action | translate}}</a>
|
||||||
|
</td>
|
||||||
|
<td class="text-center">{{item.admin}}</td>
|
||||||
|
<td class="text-center">{{item.defaultTopicPerm}}</td>
|
||||||
|
<td class="text-center">{{item.defaultGroupPerm}}</td>
|
||||||
|
<td class="text-center">
|
||||||
|
<table ng-repeat="topic in item.topicPerms" class="perm-table">
|
||||||
|
<tr>
|
||||||
|
<td class="perm-tg">{{topic}}</td>
|
||||||
|
<td ng-show="writeOperationEnabled" class="center"><a
|
||||||
|
href="javascript:;"
|
||||||
|
ng-click="openUpdateTopicDialog(item, topic)">
|
||||||
|
{{'UPDATE' | translate}}</a></td>
|
||||||
|
<td ng-show="writeOperationEnabled"><a href="javascript:;"
|
||||||
|
ng-confirm-click="Are you sure to delete {{topic}}?"
|
||||||
|
confirmed-click="deletePermConfig(item, topic, 'topic')">{{'DELETE' | translate}}</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</td>
|
||||||
|
<td class="text-center">
|
||||||
|
<table ng-repeat="group in item.groupPerms" class="perm-table">
|
||||||
|
<tr>
|
||||||
|
<td class="perm-tg">{{group}}</td>
|
||||||
|
<td ng-show="writeOperationEnabled" class="center"><a
|
||||||
|
href="javascript:;"
|
||||||
|
ng-click="openUpdateGroupDialog(item, group)">
|
||||||
|
{{'UPDATE' | translate}}</a></td>
|
||||||
|
<td ng-show="writeOperationEnabled"><a href="javascript:;"
|
||||||
|
ng-confirm-click="Are you sure to delete {{group}}?"
|
||||||
|
confirmed-click="deletePermConfig(item, group, 'group')">{{'DELETE' | translate}}</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</td>
|
||||||
|
<td ng-show="writeOperationEnabled" class="text-center">
|
||||||
|
<button class="btn btn-raised btn-sm btn-primary" type="button"
|
||||||
|
ng-click="openAddTopicDialog(item)">
|
||||||
|
{{'ADD' | translate}}topic
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-raised btn-sm btn-primary" type="button"
|
||||||
|
ng-click="openAddGroupDialog(item)">
|
||||||
|
{{'ADD' | translate}}group
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-raised btn-sm btn-primary" type="button"
|
||||||
|
ng-click="openUpdateDialog(item)">
|
||||||
|
{{'UPDATE' | translate}}
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-raised btn-sm btn-danger" type="button"
|
||||||
|
ng-confirm-click="Are you sure to delete {{item.accessKey}}?"
|
||||||
|
confirmed-click="deleteAclConfig(item.accessKey)">{{'DELETE' | translate}}
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-raised btn-sm btn-danger" type="button"
|
||||||
|
ng-click="synchronizeData(item)">{{'SYNCHRONIZE' | translate}}
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
<tm-pagination conf="paginationConf"></tm-pagination>
|
||||||
|
</md-content>
|
||||||
|
</md-tab>
|
||||||
|
<md-tab label="Global White List">
|
||||||
|
<md-content class="md-padding" style="min-height:600px">
|
||||||
|
<form class="form-inline pull-left col-sm-12">
|
||||||
|
<button ng-show="writeOperationEnabled" class="btn btn-raised btn-sm btn-primary"
|
||||||
|
type="button"
|
||||||
|
ng-click="openAddAddrDialog()">{{'ADD' |
|
||||||
|
translate}}
|
||||||
|
</button>
|
||||||
|
<button ng-show="writeOperationEnabled" class="btn btn-raised btn-sm btn-danger"
|
||||||
|
type="button"
|
||||||
|
ng-confirm-click="Are you sure to synchronize white list to all broker int the cluster?"
|
||||||
|
confirmed-click="synchronizeWhiteList(allGlobalWhiteAddrs)">
|
||||||
|
{{'SYNCHRONIZE' | translate}}
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
<table class="table table-bordered">
|
||||||
|
<tr>
|
||||||
|
<th class="text-center">{{'WHITE_LIST' | translate}}</th>
|
||||||
|
<th ng-show="writeOperationEnabled" class="text-center">
|
||||||
|
{{'OPERATION' | translate}}
|
||||||
|
</th>
|
||||||
|
</tr>
|
||||||
|
<tr ng-repeat="item in allGlobalWhiteAddrs">
|
||||||
|
<td class="text-center">{{item}}
|
||||||
|
</td>
|
||||||
|
<td ng-show="writeOperationEnabled" class="text-center">
|
||||||
|
<button class="btn btn-raised btn-sm btn-danger" type="button"
|
||||||
|
ng-confirm-click="Are you sure to delete {{item}}?"
|
||||||
|
confirmed-click="deleteGlobalWhiteAddr(item)">{{'DELETE' | translate}}
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</md-content>
|
||||||
|
</md-tab>
|
||||||
|
</md-tabs>
|
||||||
|
</md-content>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/ng-template" id="addAclAccountDialog">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h4 class="modal-title">{{'ADD' | translate }}</h4>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body ">
|
||||||
|
<form id="addAppForm" name="addAppForm" class="form-horizontal" novalidate>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Access Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.accessKey" type="text" required/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Secret Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.secretKey" type="text" required/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'IS_ADMIN' | translate}}:</label>
|
||||||
|
<div class="col-sm-8">
|
||||||
|
<md-switch class="md-primary" md-no-ink aria-label="Switch No Ink"
|
||||||
|
ng-model="ngDialogData.admin">
|
||||||
|
</md-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'DEFAULT_TOPIC_PERM' | translate}}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.defaultTopicPerm" type="text" readonly/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'DEFAULT_GROUP_PERM' | translate}}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.defaultGroupPerm" type="text" readonly/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="ngdialog-buttons">
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-primary"
|
||||||
|
ng-click="addRequest({'accessKey':ngDialogData.accessKey, 'secretKey': ngDialogData.secretKey, 'admin': ngDialogData.admin, 'defaultTopicPerm': ngDialogData.defaultTopicPerm, 'defaultGroupPerm': ngDialogData.defaultGroupPerm})">
|
||||||
|
{{ 'COMMIT' | translate }}
|
||||||
|
</button>
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-secondary"
|
||||||
|
ng-click="closeThisDialog('Cancel')">{{ 'CLOSE' | translate }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/ng-template" id="updateAclAccountDialog">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h4 class="modal-title">{{'UPDATE' | translate }}</h4>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body ">
|
||||||
|
<form id="updateAccountForm" name="updateAccountForm" class="form-horizontal" novalidate>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Access Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.accessKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Secret Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.secretKey" type="text" required/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'IS_ADMIN' | translate}}:</label>
|
||||||
|
<div class="col-sm-8">
|
||||||
|
<md-switch class="md-primary" md-no-ink aria-label="Switch No Ink"
|
||||||
|
ng-model="ngDialogData.admin">
|
||||||
|
</md-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'DEFAULT_TOPIC_PERM' | translate}}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.defaultTopicPerm" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'DEFAULT_GROUP_PERM' | translate}}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.defaultGroupPerm" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="ngdialog-buttons">
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-primary"
|
||||||
|
ng-click="updateAclAccountRequest({'accessKey':ngDialogData.accessKey, 'secretKey': ngDialogData.secretKey, 'admin': ngDialogData.admin, 'defaultTopicPerm': ngDialogData.defaultTopicPerm, 'defaultGroupPerm': ngDialogData.defaultGroupPerm})">
|
||||||
|
{{ 'COMMIT' | translate }}
|
||||||
|
</button>
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-secondary"
|
||||||
|
ng-click="closeThisDialog('Cancel')">{{ 'CLOSE' | translate }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/ng-template" id="addAclTopicDialog">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h4 class="modal-title">{{'ADD' | translate }}{{'TOPIC_PERM' | translate }}</h4>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body ">
|
||||||
|
<form id="addAclTopicForm" name="addAclTopicForm" class="form-horizontal" novalidate>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Access Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.accessKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Secret Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.secretKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'TOPIC' | translate}}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="topic" type="text" required/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'TOPIC_PERM' | translate}}:</label>
|
||||||
|
<div class="col-sm-8">
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.pub">PUB</md-checkbox>
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.sub">SUB</md-checkbox>
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.deny">DENY</md-checkbox>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="ngdialog-buttons">
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-primary"
|
||||||
|
ng-click="updateAclAccountRequest({'originalData':ngDialogData,'topic': topic , 'pub': ngDialogData.pub, 'sub': ngDialogData.sub, 'deny': ngDialogData.deny})">
|
||||||
|
{{ 'COMMIT' | translate }}
|
||||||
|
</button>
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-secondary"
|
||||||
|
ng-click="closeThisDialog('Cancel')">{{ 'CLOSE' | translate }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/ng-template" id="updateAclTopicDialog">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h4 class="modal-title">{{'UPDATE' | translate }}{{'TOPIC_PERM' | translate }}</h4>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body ">
|
||||||
|
<form id="updateAclTopicForm" name="updateAclTopicForm" class="form-horizontal" novalidate>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Access Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.accessKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Secret Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.secretKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'TOPIC' | translate}}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.topic" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'TOPIC_PERM' | translate}}:</label>
|
||||||
|
<div class="col-sm-8">
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.pub">PUB</md-checkbox>
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.sub">SUB</md-checkbox>
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.deny">DENY</md-checkbox>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="ngdialog-buttons">
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-primary"
|
||||||
|
ng-click="updateAclAccountRequest({'originalData':ngDialogData,'topic': ngDialogData.topic , 'pub': ngDialogData.pub, 'sub': ngDialogData.sub, 'deny': ngDialogData.deny})">
|
||||||
|
{{ 'COMMIT' | translate }}
|
||||||
|
</button>
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-secondary"
|
||||||
|
ng-click="closeThisDialog('Cancel')">{{ 'CLOSE' | translate }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/ng-template" id="addAclGroupDialog">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h4 class="modal-title">{{'ADD' | translate }}{{'GROUP_PERM' | translate }}</h4>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body ">
|
||||||
|
<form id="addAclGroupForm" name="addAclGroupForm" class="form-horizontal" novalidate>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Access Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.accessKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Secret Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.secretKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'CONSUMER' | translate}}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="group" type="text" required/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'GROUP_PERM' | translate}}:</label>
|
||||||
|
<div class="col-sm-8">
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.pub">PUB</md-checkbox>
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.sub">SUB</md-checkbox>
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.deny">DENY</md-checkbox>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="ngdialog-buttons">
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-primary"
|
||||||
|
ng-click="updateAclAccountRequest({'originalData':ngDialogData,'group': group , 'pub': ngDialogData.pub, 'sub': ngDialogData.sub, 'deny': ngDialogData.deny})">
|
||||||
|
{{ 'COMMIT' | translate }}
|
||||||
|
</button>
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-secondary"
|
||||||
|
ng-click="closeThisDialog('Cancel')">{{ 'CLOSE' | translate }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/ng-template" id="updateAclGroupDialog">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h4 class="modal-title">{{'UPDATE' | translate }}{{'GROUP_PERM' | translate }}</h4>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body ">
|
||||||
|
<form id="updateAclGroupForm" name="updateAclGroupForm" class="form-horizontal" novalidate>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Access Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.accessKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">Secret Key:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.secretKey" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'CONSUMER' | translate}}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ngDialogData.group" type="text" disabled/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'GROUP_PERM' | translate}}:</label>
|
||||||
|
<div class="col-sm-8">
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.pub">PUB</md-checkbox>
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.sub">SUB</md-checkbox>
|
||||||
|
<md-checkbox class="md-primary" ng-model="ngDialogData.deny">DENY</md-checkbox>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="ngdialog-buttons">
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-primary"
|
||||||
|
ng-click="updateAclAccountRequest({'originalData':ngDialogData,'group': ngDialogData.group , 'pub': ngDialogData.pub, 'sub': ngDialogData.sub, 'deny': ngDialogData.deny})">
|
||||||
|
{{ 'COMMIT' | translate }}
|
||||||
|
</button>
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-secondary"
|
||||||
|
ng-click="closeThisDialog('Cancel')">{{ 'CLOSE' | translate }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/ng-template" id="addWhiteListDialog">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h4 class="modal-title">{{'ADD' | translate }}{{'WHITE_LIST' | translate }}</h4>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body ">
|
||||||
|
<form id="addWhiteListForm" name="addWhiteListForm" class="form-horizontal" novalidate>
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="control-label col-sm-2">{{'WHITE_LIST' | translate }}:</label>
|
||||||
|
<div class="col-sm-10">
|
||||||
|
<input class="form-control" ng-model="ip" type="text"/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="ngdialog-buttons">
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-primary"
|
||||||
|
ng-click="addWhiteListRequest(ip)">
|
||||||
|
{{ 'COMMIT' | translate }}
|
||||||
|
</button>
|
||||||
|
<button type="button" class="ngdialog-button ngdialog-button-secondary"
|
||||||
|
ng-click="closeThisDialog('Cancel')">{{ 'CLOSE' | translate }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</script>
|
@@ -64,9 +64,22 @@
|
|||||||
ng-click="queryDlqMessageByConsumerGroup()">
|
ng-click="queryDlqMessageByConsumerGroup()">
|
||||||
<span class="glyphicon glyphicon-search"></span> {{ 'SEARCH' | translate}}
|
<span class="glyphicon glyphicon-search"></span> {{ 'SEARCH' | translate}}
|
||||||
</button>
|
</button>
|
||||||
|
<button type="button" id="batchResendBtn"
|
||||||
|
class="btn btn-raised btn-sm btn-primary disabled"
|
||||||
|
data-toggle="modal"
|
||||||
|
ng-click="batchResendDlqMessage(selectedConsumerGroup)">
|
||||||
|
<span class="glyphicon glyphicon-send"></span> {{ 'BATCH_RESEND' | translate}}
|
||||||
|
</button>
|
||||||
|
<button type="button" id="batchExportBtn"
|
||||||
|
class="btn btn-raised btn-sm btn-primary disabled"
|
||||||
|
data-toggle="modal"
|
||||||
|
ng-click="batchExportDlqMessage(selectedConsumerGroup)">
|
||||||
|
<span class="glyphicon glyphicon-export"></span> {{ 'BATCH_EXPORT' | translate}}
|
||||||
|
</button>
|
||||||
</form>
|
</form>
|
||||||
<table class="table table-bordered text-middle">
|
<table class="table table-bordered text-middle">
|
||||||
<tr>
|
<tr>
|
||||||
|
<th class="text-center"><input type="checkbox" ng-model='checkedAll' ng-change="selectAll()"/></th>
|
||||||
<th class="text-center">Message ID</th>
|
<th class="text-center">Message ID</th>
|
||||||
<th class="text-center">Tag</th>
|
<th class="text-center">Tag</th>
|
||||||
<th class="text-center">Key</th>
|
<th class="text-center">Key</th>
|
||||||
@@ -77,6 +90,9 @@
|
|||||||
<td colspan="6" style="text-align: center">{{'NO_MATCH_RESULT' | translate}}</td>
|
<td colspan="6" style="text-align: center">{{'NO_MATCH_RESULT' | translate}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr ng-repeat="item in messageShowList">
|
<tr ng-repeat="item in messageShowList">
|
||||||
|
<td class="text-center">
|
||||||
|
<input type="checkbox" ng-model='item.checked' ng-change="selectItem()"/>
|
||||||
|
</td>
|
||||||
<td class="text-center">{{item.msgId}}</td>
|
<td class="text-center">{{item.msgId}}</td>
|
||||||
<td class="text-center">{{item.properties.TAGS}}</td>
|
<td class="text-center">{{item.properties.TAGS}}</td>
|
||||||
<td class="text-center">{{item.properties.KEYS}}</td>
|
<td class="text-center">{{item.properties.KEYS}}</td>
|
||||||
@@ -192,7 +208,7 @@
|
|||||||
<label class="control-label col-sm-2">Properties:</label>
|
<label class="control-label col-sm-2">Properties:</label>
|
||||||
<div class="col-sm-10">
|
<div class="col-sm-10">
|
||||||
<textarea class="form-control"
|
<textarea class="form-control"
|
||||||
style="min-height:60px; resize: none"
|
style="min-height:100px; resize: none"
|
||||||
readonly>{{ngDialogData.messageView.properties}}</textarea>
|
readonly>{{ngDialogData.messageView.properties}}</textarea>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@@ -73,6 +73,9 @@
|
|||||||
<th class="text-center">StoreTime</th>
|
<th class="text-center">StoreTime</th>
|
||||||
<th class="text-center">Operation</th>
|
<th class="text-center">Operation</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr style="display: none" id="noMsgTip">
|
||||||
|
<td colspan="5" style="text-align: center">{{'NO_MATCH_RESULT' | translate}}</td>
|
||||||
|
</tr>
|
||||||
<tr ng-repeat="item in messageShowList">
|
<tr ng-repeat="item in messageShowList">
|
||||||
<td class="text-center">{{item.msgId}}</td>
|
<td class="text-center">{{item.msgId}}</td>
|
||||||
<td class="text-center">{{item.properties.TAGS}}</td>
|
<td class="text-center">{{item.properties.TAGS}}</td>
|
||||||
@@ -227,7 +230,13 @@
|
|||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
<p>messageTrackList:</p>
|
<div style="font-weight:700; color:#000">messageTrackList:</div>
|
||||||
|
<form class="form-inline">
|
||||||
|
<div class="form-group">
|
||||||
|
<label>consumerGroup:</label>
|
||||||
|
<input type="text" class="form-control" ng-model="filterConsumerGroup">
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
<table class="table-bordered table text-middle">
|
<table class="table-bordered table text-middle">
|
||||||
<tr>
|
<tr>
|
||||||
<th class="text-center">consumerGroup</th>
|
<th class="text-center">consumerGroup</th>
|
||||||
@@ -235,7 +244,7 @@
|
|||||||
<!--<th class="text-center">exceptionDesc</th>-->
|
<!--<th class="text-center">exceptionDesc</th>-->
|
||||||
<th class="text-center">Operation</th>
|
<th class="text-center">Operation</th>
|
||||||
</tr>
|
</tr>
|
||||||
<tr ng-repeat="item in ngDialogData.messageTrackList">
|
<tr ng-repeat="item in messageTrackShowList">
|
||||||
<td class="text-center">{{item.consumerGroup}}</td>
|
<td class="text-center">{{item.consumerGroup}}</td>
|
||||||
<td class="text-center">{{item.trackType}}</td>
|
<td class="text-center">{{item.trackType}}</td>
|
||||||
<td class="text-center">
|
<td class="text-center">
|
||||||
|
@@ -17,15 +17,26 @@
|
|||||||
<div class="container-fluid" id="deployHistoryList">
|
<div class="container-fluid" id="deployHistoryList">
|
||||||
<div class="page-content">
|
<div class="page-content">
|
||||||
<h2 class="md-title">NameServerAddressList</h2>
|
<h2 class="md-title">NameServerAddressList</h2>
|
||||||
<div class="pull-left">
|
<div class="pull-left" style="min-width: 400px; max-width: 500px; padding: 10px 10px 10px 0">
|
||||||
<md-chips ng-model="namesvrAddrList" md-on-add="eleChange(namesvrAddrList)"
|
<select ng-model="selectedNamesrv" chosen
|
||||||
md-on-remove="eleChange(namesvrAddrList)" readonly="inputReadonly" md-removable="ctrl.removable"></md-chips>
|
ng-options="x for x in namesvrAddrList"
|
||||||
|
ng-change="updateNameSvrAddr()"
|
||||||
|
required></select>
|
||||||
</div>
|
</div>
|
||||||
<div class="pull-left">
|
<div class="pull-left">
|
||||||
<button class="btn btn-raised btn-sm btn-primary" type="button" ng-show="{{writeOperationEnabled}}"
|
<button class="btn btn-raised btn-sm btn-primary" type="button" ng-show="{{writeOperationEnabled}}"
|
||||||
ng-click="updateNameSvrAddr()">{{'UPDATE' | translate}}
|
ng-click="updateNameSvrAddr()">{{'UPDATE' | translate}}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
<form class="form-inline pull-left" style="margin-left: 20px" ng-show="{{writeOperationEnabled}}">
|
||||||
|
<div class="form-group" style="margin: 0">
|
||||||
|
<label for="namesrvAddr">NamesrvAddr:</label>
|
||||||
|
<input id="namesrvAddr" class="form-control" style="width: 300px; margin: 0 10px 0 10px" type="text" ng-model="newNamesrvAddr" required/>
|
||||||
|
<button class="btn btn-raised btn-sm btn-primary" type="button"
|
||||||
|
ng-click="addNameSvrAddr()"> {{ 'ADD' | translate}}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
<br/>
|
<br/>
|
||||||
<br/>
|
<br/>
|
||||||
<h2 class="md-title">IsUseVIPChannel</h2>
|
<h2 class="md-title">IsUseVIPChannel</h2>
|
||||||
|
@@ -418,8 +418,10 @@ public class MQAdminExtImplTest {
|
|||||||
assertNotNull(mqAdminExtImpl);
|
assertNotNull(mqAdminExtImpl);
|
||||||
{
|
{
|
||||||
doNothing().when(defaultMQAdminExt).deleteSubscriptionGroup(anyString(), anyString());
|
doNothing().when(defaultMQAdminExt).deleteSubscriptionGroup(anyString(), anyString());
|
||||||
|
doNothing().when(defaultMQAdminExt).deleteSubscriptionGroup(anyString(), anyString(), anyBoolean());
|
||||||
}
|
}
|
||||||
mqAdminExtImpl.deleteSubscriptionGroup(brokerAddr, "group_test");
|
mqAdminExtImpl.deleteSubscriptionGroup(brokerAddr, "group_test");
|
||||||
|
mqAdminExtImpl.deleteSubscriptionGroup(brokerAddr, "group_test", true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@@ -758,16 +760,6 @@ public class MQAdminExtImplTest {
|
|||||||
Assert.assertNotNull(wrapper);
|
Assert.assertNotNull(wrapper);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testGetAllTopicGroup() throws Exception {
|
|
||||||
assertNotNull(mqAdminExtImpl);
|
|
||||||
{
|
|
||||||
when(defaultMQAdminExt.getAllTopicGroup(anyString(), anyLong())).thenReturn(new TopicConfigSerializeWrapper());
|
|
||||||
}
|
|
||||||
TopicConfigSerializeWrapper wrapper = mqAdminExtImpl.getAllTopicGroup(brokerAddr, 5000L);
|
|
||||||
Assert.assertNotNull(wrapper);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testUpdateConsumeOffset() throws Exception {
|
public void testUpdateConsumeOffset() throws Exception {
|
||||||
assertNotNull(mqAdminExtImpl);
|
assertNotNull(mqAdminExtImpl);
|
||||||
@@ -801,4 +793,42 @@ public class MQAdminExtImplTest {
|
|||||||
Assert.assertFalse(mqAdminExtImpl.resumeCheckHalfMessage("topic_test", "7F000001ACC018B4AAC2116AF6500000"));
|
Assert.assertFalse(mqAdminExtImpl.resumeCheckHalfMessage("topic_test", "7F000001ACC018B4AAC2116AF6500000"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAddWritePermOfBroker() throws Exception {
|
||||||
|
assertNotNull(mqAdminExtImpl);
|
||||||
|
{
|
||||||
|
when(defaultMQAdminExt.addWritePermOfBroker(anyString(), anyString())).thenReturn(6);
|
||||||
|
}
|
||||||
|
Assert.assertEquals(mqAdminExtImpl.addWritePermOfBroker("127.0.0.1:9876", "broker-a"), 6);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetUserSubscriptionGroup() throws Exception {
|
||||||
|
assertNotNull(mqAdminExtImpl);
|
||||||
|
SubscriptionGroupWrapper wrapper = new SubscriptionGroupWrapper();
|
||||||
|
{
|
||||||
|
when(defaultMQAdminExt.getUserSubscriptionGroup(anyString(), anyLong())).thenReturn(wrapper);
|
||||||
|
}
|
||||||
|
Assert.assertEquals(mqAdminExtImpl.getUserSubscriptionGroup("127.0.0.1:10911", 3000), wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetAllTopicConfig() throws Exception {
|
||||||
|
assertNotNull(mqAdminExtImpl);
|
||||||
|
TopicConfigSerializeWrapper wrapper = new TopicConfigSerializeWrapper();
|
||||||
|
{
|
||||||
|
when(defaultMQAdminExt.getAllTopicConfig(anyString(), anyLong())).thenReturn(wrapper);
|
||||||
|
}
|
||||||
|
Assert.assertEquals(mqAdminExtImpl.getAllTopicConfig("127.0.0.1:10911", 3000), wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetUserTopicConfig() throws Exception {
|
||||||
|
assertNotNull(mqAdminExtImpl);
|
||||||
|
TopicConfigSerializeWrapper wrapper = new TopicConfigSerializeWrapper();
|
||||||
|
{
|
||||||
|
when(defaultMQAdminExt.getUserTopicConfig(anyString(), anyBoolean(), anyLong())).thenReturn(wrapper);
|
||||||
|
}
|
||||||
|
Assert.assertEquals(mqAdminExtImpl.getUserTopicConfig("127.0.0.1:10911", true, 3000), wrapper);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,50 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.config;
|
||||||
|
|
||||||
|
import java.util.concurrent.CountDownLatch;
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class CollectExecutorConfigTest {
|
||||||
|
|
||||||
|
private final static int COUNT = 100;
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCollectExecutor() throws Exception {
|
||||||
|
AtomicInteger num = new AtomicInteger(0);
|
||||||
|
CollectExecutorConfig config = new CollectExecutorConfig();
|
||||||
|
config.setCoreSize(10);
|
||||||
|
config.setMaxSize(10);
|
||||||
|
config.setQueueSize(500);
|
||||||
|
config.setKeepAliveTime(3000);
|
||||||
|
ExecutorService collectExecutor = config.collectExecutor(config);
|
||||||
|
Assert.assertNotNull(collectExecutor);
|
||||||
|
CountDownLatch countDownLatch = new CountDownLatch(COUNT);
|
||||||
|
for (int i = 0; i < COUNT; i++) {
|
||||||
|
collectExecutor.submit(() -> {
|
||||||
|
num.getAndIncrement();
|
||||||
|
countDownLatch.countDown();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
countDownLatch.await();
|
||||||
|
System.out.println(collectExecutor.isTerminated());
|
||||||
|
Assert.assertEquals(COUNT, num.get());
|
||||||
|
}
|
||||||
|
}
|
@@ -17,6 +17,7 @@
|
|||||||
|
|
||||||
package org.apache.rocketmq.dashboard.config;
|
package org.apache.rocketmq.dashboard.config;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
@@ -39,6 +40,7 @@ public class RMQConfigureTest {
|
|||||||
rmqConfigure.setLoginRequired(true);
|
rmqConfigure.setLoginRequired(true);
|
||||||
rmqConfigure.setNamesrvAddr("127.0.0.1:9876");
|
rmqConfigure.setNamesrvAddr("127.0.0.1:9876");
|
||||||
rmqConfigure.setTimeoutMillis(3000L);
|
rmqConfigure.setTimeoutMillis(3000L);
|
||||||
|
rmqConfigure.setNamesrvAddrs(Lists.asList("127.0.0.1:9876", new String[] {"127.0.0.2:9876"}));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@@ -54,6 +56,7 @@ public class RMQConfigureTest {
|
|||||||
Assert.assertTrue(rmqConfigure.isEnableDashBoardCollect());
|
Assert.assertTrue(rmqConfigure.isEnableDashBoardCollect());
|
||||||
Assert.assertTrue(rmqConfigure.isLoginRequired());
|
Assert.assertTrue(rmqConfigure.isLoginRequired());
|
||||||
Assert.assertEquals(rmqConfigure.getNamesrvAddr(), "127.0.0.1:9876");
|
Assert.assertEquals(rmqConfigure.getNamesrvAddr(), "127.0.0.1:9876");
|
||||||
|
Assert.assertEquals(rmqConfigure.getNamesrvAddrs().size(), 2);
|
||||||
Assert.assertEquals(rmqConfigure.getTimeoutMillis().longValue(), 3000L);
|
Assert.assertEquals(rmqConfigure.getTimeoutMillis().longValue(), 3000L);
|
||||||
ErrorPageRegistrar registrar = rmqConfigure.errorPageRegistrar();
|
ErrorPageRegistrar registrar = rmqConfigure.errorPageRegistrar();
|
||||||
registrar.registerErrorPages(new ErrorPageRegistry() {
|
registrar.registerErrorPages(new ErrorPageRegistry() {
|
||||||
|
@@ -0,0 +1,368 @@
|
|||||||
|
/*
|
||||||
|
* 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.dashboard.controller;
|
||||||
|
|
||||||
|
import com.alibaba.fastjson.JSON;
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
import java.util.List;
|
||||||
|
import org.apache.rocketmq.common.AclConfig;
|
||||||
|
import org.apache.rocketmq.common.PlainAccessConfig;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.ClusterInfo;
|
||||||
|
import org.apache.rocketmq.dashboard.model.request.AclRequest;
|
||||||
|
import org.apache.rocketmq.dashboard.service.impl.AclServiceImpl;
|
||||||
|
import org.apache.rocketmq.dashboard.util.MockObjectUtil;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.mockito.InjectMocks;
|
||||||
|
import org.mockito.Spy;
|
||||||
|
import org.springframework.http.MediaType;
|
||||||
|
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders;
|
||||||
|
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.ArgumentMatchers.anyString;
|
||||||
|
import static org.mockito.Mockito.doNothing;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
|
||||||
|
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;
|
||||||
|
|
||||||
|
public class AclControllerTest extends BaseControllerTest {
|
||||||
|
|
||||||
|
@InjectMocks
|
||||||
|
private AclController aclController;
|
||||||
|
|
||||||
|
@Spy
|
||||||
|
private AclServiceImpl aclService;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void init() throws Exception {
|
||||||
|
AclConfig aclConfig = MockObjectUtil.createAclConfig();
|
||||||
|
when(mqAdminExt.examineBrokerClusterAclConfig(anyString())).thenReturn(aclConfig);
|
||||||
|
ClusterInfo clusterInfo = MockObjectUtil.createClusterInfo();
|
||||||
|
when(mqAdminExt.examineBrokerClusterInfo()).thenReturn(clusterInfo);
|
||||||
|
doNothing().when(mqAdminExt).createAndUpdatePlainAccessConfig(anyString(), any(PlainAccessConfig.class));
|
||||||
|
doNothing().when(mqAdminExt).deletePlainAccessConfig(anyString(), anyString());
|
||||||
|
doNothing().when(mqAdminExt).updateGlobalWhiteAddrConfig(anyString(), anyString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testIsEnableAcl() throws Exception {
|
||||||
|
final String url = "/acl/enable.query";
|
||||||
|
// 1. disable acl.
|
||||||
|
requestBuilder = MockMvcRequestBuilders.get(url);
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.data").value(false));
|
||||||
|
|
||||||
|
// 2.enable acl.
|
||||||
|
super.mockRmqConfigure();
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.data").value(true));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetAclConfig() throws Exception {
|
||||||
|
final String url = "/acl/config.query";
|
||||||
|
|
||||||
|
// 1. broker addr table is not empty.
|
||||||
|
ClusterInfo clusterInfo = MockObjectUtil.createClusterInfo();
|
||||||
|
when(mqAdminExt.examineBrokerClusterInfo()).thenReturn(clusterInfo);
|
||||||
|
requestBuilder = MockMvcRequestBuilders.get(url);
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.data").isMap())
|
||||||
|
.andExpect(jsonPath("$.data.globalWhiteAddrs").isNotEmpty())
|
||||||
|
.andExpect(jsonPath("$.data.plainAccessConfigs").isNotEmpty())
|
||||||
|
.andExpect(jsonPath("$.data.plainAccessConfigs[0].secretKey").isNotEmpty());
|
||||||
|
|
||||||
|
// 2. broker addr table is empty.
|
||||||
|
clusterInfo.getBrokerAddrTable().clear();
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.data").isMap())
|
||||||
|
.andExpect(jsonPath("$.data.globalWhiteAddrs").isEmpty())
|
||||||
|
.andExpect(jsonPath("$.data.plainAccessConfigs").isEmpty());
|
||||||
|
|
||||||
|
// 3. login required and user info is null.
|
||||||
|
when(configure.isLoginRequired()).thenReturn(true);
|
||||||
|
when(mqAdminExt.examineBrokerClusterInfo()).thenReturn(MockObjectUtil.createClusterInfo());
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.data").isMap())
|
||||||
|
.andExpect(jsonPath("$.data.globalWhiteAddrs").isNotEmpty())
|
||||||
|
.andExpect(jsonPath("$.data.plainAccessConfigs").isNotEmpty())
|
||||||
|
.andExpect(jsonPath("$.data.plainAccessConfigs[0].secretKey").isEmpty());
|
||||||
|
// 4. login required, but user is not admin. emmmm, Mockito may can not mock static method.
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAddAclConfig() throws Exception {
|
||||||
|
final String url = "/acl/add.do";
|
||||||
|
PlainAccessConfig accessConfig = new PlainAccessConfig();
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
|
||||||
|
// 1. access key is null.
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(-1))
|
||||||
|
.andExpect(jsonPath("$.errMsg").exists());
|
||||||
|
|
||||||
|
// 2. secret key is null.
|
||||||
|
accessConfig.setAccessKey("test-access-key");
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(-1))
|
||||||
|
.andExpect(jsonPath("$.errMsg").exists());
|
||||||
|
|
||||||
|
ClusterInfo clusterInfo = MockObjectUtil.createClusterInfo();
|
||||||
|
when(mqAdminExt.examineBrokerClusterInfo()).thenReturn(clusterInfo);
|
||||||
|
|
||||||
|
// 3. add if the access key not exist.
|
||||||
|
accessConfig.setSecretKey("12345678");
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
|
||||||
|
// 4. add failed if the access key is existed.
|
||||||
|
accessConfig.setAccessKey("rocketmq2");
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(-1))
|
||||||
|
.andExpect(jsonPath("$.errMsg").exists());
|
||||||
|
|
||||||
|
// 5. add failed if there is no alive broker.
|
||||||
|
clusterInfo.getBrokerAddrTable().clear();
|
||||||
|
accessConfig.setAccessKey("test-access-key");
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(-1))
|
||||||
|
.andExpect(jsonPath("$.errMsg").exists());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDeleteAclConfig() throws Exception {
|
||||||
|
final String url = "/acl/delete.do";
|
||||||
|
PlainAccessConfig accessConfig = new PlainAccessConfig();
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
|
||||||
|
// 1. access key is null.
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(-1))
|
||||||
|
.andExpect(jsonPath("$.errMsg").exists());
|
||||||
|
|
||||||
|
// 2. access key is not null.
|
||||||
|
accessConfig.setAccessKey("rocketmq");
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUpdateAclConfig() throws Exception {
|
||||||
|
final String url = "/acl/update.do";
|
||||||
|
PlainAccessConfig accessConfig = new PlainAccessConfig();
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
|
||||||
|
// 1. secret key is null.
|
||||||
|
accessConfig.setAccessKey("rocketmq");
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(-1))
|
||||||
|
.andExpect(jsonPath("$.errMsg").exists());
|
||||||
|
|
||||||
|
// 2. update.
|
||||||
|
accessConfig.setSecretKey("abcdefghjkl");
|
||||||
|
requestBuilder.content(JSON.toJSONString(accessConfig));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAddAclTopicConfig() throws Exception {
|
||||||
|
final String url = "/acl/topic/add.do";
|
||||||
|
AclRequest request = new AclRequest();
|
||||||
|
request.setConfig(createDefaultPlainAccessConfig());
|
||||||
|
|
||||||
|
// 1. if not exist.
|
||||||
|
request.setTopicPerm("test_topic=PUB");
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
requestBuilder.content(JSON.toJSONString(request));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
|
||||||
|
// 2. if exist.
|
||||||
|
request.setTopicPerm("topicA=PUB");
|
||||||
|
requestBuilder.content(JSON.toJSONString(request));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
|
||||||
|
// 3. if access key not exist.
|
||||||
|
request.getConfig().setAccessKey("test_access_key123");
|
||||||
|
requestBuilder.content(JSON.toJSONString(request));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAddAclGroupConfig() throws Exception {
|
||||||
|
final String url = "/acl/group/add.do";
|
||||||
|
AclRequest request = new AclRequest();
|
||||||
|
request.setConfig(createDefaultPlainAccessConfig());
|
||||||
|
|
||||||
|
// 1. if not exist.
|
||||||
|
request.setGroupPerm("test_consumer=PUB|SUB");
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
requestBuilder.content(JSON.toJSONString(request));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
|
||||||
|
// 2. if exist.
|
||||||
|
request.setGroupPerm("groupA=PUB|SUB");
|
||||||
|
requestBuilder.content(JSON.toJSONString(request));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
|
||||||
|
// 3. if access key not exist.
|
||||||
|
request.getConfig().setAccessKey("test_access_key123");
|
||||||
|
requestBuilder.content(JSON.toJSONString(request));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDeletePermConfig() throws Exception {
|
||||||
|
final String url = "/acl/perm/delete.do";
|
||||||
|
AclRequest request = new AclRequest();
|
||||||
|
request.setConfig(createDefaultPlainAccessConfig());
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
requestBuilder.content(JSON.toJSONString(request));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
|
||||||
|
// if access key not exist.
|
||||||
|
request.getConfig().setAccessKey("test_access_key123");
|
||||||
|
requestBuilder.content(JSON.toJSONString(request));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSyncConfig() throws Exception {
|
||||||
|
final String url = "/acl/sync.do";
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
requestBuilder.content(JSON.toJSONString(createDefaultPlainAccessConfig()));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAddWhiteList() throws Exception {
|
||||||
|
final String url = "/acl/white/list/add.do";
|
||||||
|
List<String> whiteList = Lists.newArrayList("192.168.0.1");
|
||||||
|
|
||||||
|
// 1. if global white list is not null.
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
requestBuilder.content(JSON.toJSONString(whiteList));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
|
||||||
|
// 2. if global white list is null.
|
||||||
|
AclConfig aclConfig = MockObjectUtil.createAclConfig();
|
||||||
|
aclConfig.setGlobalWhiteAddrs(null);
|
||||||
|
when(mqAdminExt.examineBrokerClusterAclConfig(anyString())).thenReturn(aclConfig);
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDeleteWhiteAddr() throws Exception {
|
||||||
|
final String url = "/acl/white/list/delete.do";
|
||||||
|
requestBuilder = MockMvcRequestBuilders.delete(url);
|
||||||
|
requestBuilder.param("request", "localhost");
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSynchronizeWhiteList() throws Exception {
|
||||||
|
final String url = "/acl/white/list/sync.do";
|
||||||
|
List<String> whiteList = Lists.newArrayList();
|
||||||
|
|
||||||
|
// 1. if white list for syncing is empty.
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
requestBuilder.content(JSON.toJSONString(whiteList));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(-1))
|
||||||
|
.andExpect(jsonPath("$.errMsg").exists());
|
||||||
|
|
||||||
|
// 2. if white list for syncing is not empty.
|
||||||
|
whiteList.add("localhost");
|
||||||
|
requestBuilder.content(JSON.toJSONString(whiteList));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.status").value(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override protected Object getTestController() {
|
||||||
|
return aclController;
|
||||||
|
}
|
||||||
|
|
||||||
|
private PlainAccessConfig createDefaultPlainAccessConfig() {
|
||||||
|
PlainAccessConfig config = new PlainAccessConfig();
|
||||||
|
config.setAdmin(false);
|
||||||
|
config.setAccessKey("rocketmq");
|
||||||
|
config.setSecretKey("123456789");
|
||||||
|
config.setDefaultGroupPerm("SUB");
|
||||||
|
config.setDefaultTopicPerm("DENY");
|
||||||
|
config.setTopicPerms(Lists.newArrayList("topicA=DENY", "topicB=PUB|SUB"));
|
||||||
|
config.setGroupPerms(Lists.newArrayList("groupA=DENY", "groupB=PUB|SUB"));
|
||||||
|
|
||||||
|
return config;
|
||||||
|
}
|
||||||
|
}
|
@@ -17,6 +17,7 @@
|
|||||||
|
|
||||||
package org.apache.rocketmq.dashboard.controller;
|
package org.apache.rocketmq.dashboard.controller;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
import org.apache.rocketmq.dashboard.BaseTest;
|
import org.apache.rocketmq.dashboard.BaseTest;
|
||||||
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
||||||
import org.apache.rocketmq.dashboard.support.GlobalExceptionHandler;
|
import org.apache.rocketmq.dashboard.support.GlobalExceptionHandler;
|
||||||
@@ -62,6 +63,7 @@ public abstract class BaseControllerTest extends BaseTest {
|
|||||||
when(configure.getAccessKey()).thenReturn("12345678");
|
when(configure.getAccessKey()).thenReturn("12345678");
|
||||||
when(configure.getSecretKey()).thenReturn("rocketmq");
|
when(configure.getSecretKey()).thenReturn("rocketmq");
|
||||||
when(configure.getNamesrvAddr()).thenReturn("127.0.0.1:9876");
|
when(configure.getNamesrvAddr()).thenReturn("127.0.0.1:9876");
|
||||||
|
when(configure.getNamesrvAddrs()).thenReturn(Lists.newArrayList("127.0.0.1:9876", "127.0.0.2:9876"));
|
||||||
when(configure.isACLEnabled()).thenReturn(true);
|
when(configure.isACLEnabled()).thenReturn(true);
|
||||||
when(configure.isUseTLS()).thenReturn(false);
|
when(configure.isUseTLS()).thenReturn(false);
|
||||||
}
|
}
|
||||||
|
@@ -182,6 +182,8 @@ public class ConsumerControllerTest extends BaseControllerTest {
|
|||||||
final String url = "/consumer/deleteSubGroup.do";
|
final String url = "/consumer/deleteSubGroup.do";
|
||||||
{
|
{
|
||||||
doNothing().when(mqAdminExt).deleteSubscriptionGroup(any(), anyString());
|
doNothing().when(mqAdminExt).deleteSubscriptionGroup(any(), anyString());
|
||||||
|
doNothing().when(mqAdminExt).deleteTopicInBroker(any(), anyString());
|
||||||
|
doNothing().when(mqAdminExt).deleteTopicInNameServer(any(), anyString());
|
||||||
}
|
}
|
||||||
DeleteSubGroupRequest request = new DeleteSubGroupRequest();
|
DeleteSubGroupRequest request = new DeleteSubGroupRequest();
|
||||||
request.setBrokerNameList(Lists.newArrayList("broker-a"));
|
request.setBrokerNameList(Lists.newArrayList("broker-a"));
|
||||||
|
@@ -152,7 +152,7 @@ public class DashboardControllerTest extends BaseControllerTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testTopicCurrent() throws Exception {
|
public void testTopicCurrent() throws Exception {
|
||||||
final String url = "/dashboard/topicCurrent";
|
final String url = "/dashboard/topicCurrent.query";
|
||||||
requestBuilder = MockMvcRequestBuilders.get(url);
|
requestBuilder = MockMvcRequestBuilders.get(url);
|
||||||
perform = mockMvc.perform(requestBuilder);
|
perform = mockMvc.perform(requestBuilder);
|
||||||
perform.andExpect(status().isOk())
|
perform.andExpect(status().isOk())
|
||||||
|
@@ -18,10 +18,14 @@ package org.apache.rocketmq.dashboard.controller;
|
|||||||
|
|
||||||
import com.alibaba.fastjson.JSON;
|
import com.alibaba.fastjson.JSON;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
|
import java.util.List;
|
||||||
import org.apache.rocketmq.client.exception.MQClientException;
|
import org.apache.rocketmq.client.exception.MQClientException;
|
||||||
import org.apache.rocketmq.common.MixAll;
|
import org.apache.rocketmq.common.MixAll;
|
||||||
import org.apache.rocketmq.common.protocol.ResponseCode;
|
import org.apache.rocketmq.common.protocol.ResponseCode;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.CMResult;
|
||||||
|
import org.apache.rocketmq.common.protocol.body.ConsumeMessageDirectlyResult;
|
||||||
import org.apache.rocketmq.common.protocol.route.TopicRouteData;
|
import org.apache.rocketmq.common.protocol.route.TopicRouteData;
|
||||||
|
import org.apache.rocketmq.dashboard.model.DlqMessageRequest;
|
||||||
import org.apache.rocketmq.dashboard.model.MessagePage;
|
import org.apache.rocketmq.dashboard.model.MessagePage;
|
||||||
import org.apache.rocketmq.dashboard.model.MessageView;
|
import org.apache.rocketmq.dashboard.model.MessageView;
|
||||||
import org.apache.rocketmq.dashboard.model.request.MessageQuery;
|
import org.apache.rocketmq.dashboard.model.request.MessageQuery;
|
||||||
@@ -128,10 +132,46 @@ public class DlqMessageControllerTest extends BaseControllerTest {
|
|||||||
// 2、export dlqMessage success
|
// 2、export dlqMessage success
|
||||||
perform = mockMvc.perform(requestBuilder);
|
perform = mockMvc.perform(requestBuilder);
|
||||||
perform.andExpect(status().is(200))
|
perform.andExpect(status().is(200))
|
||||||
.andExpect(content().contentType("application/vnd.ms-excel"));
|
.andExpect(content().contentType("application/vnd.ms-excel;charset=utf-8"));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBatchResendDlqMessage() throws Exception {
|
||||||
|
final String url = "/dlqMessage/batchResendDlqMessage.do";
|
||||||
|
List<DlqMessageRequest> dlqMessages = MockObjectUtil.createDlqMessageRequest();
|
||||||
|
{
|
||||||
|
ConsumeMessageDirectlyResult result = new ConsumeMessageDirectlyResult();
|
||||||
|
result.setConsumeResult(CMResult.CR_SUCCESS);
|
||||||
|
when(messageService.consumeMessageDirectly(any(), any(), any(), any())).thenReturn(result);
|
||||||
|
}
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
requestBuilder.content(JSON.toJSONString(dlqMessages));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.data", hasSize(2)))
|
||||||
|
.andExpect(jsonPath("$.data[0].consumeResult").value("CR_SUCCESS"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBatchExportDlqMessage() throws Exception {
|
||||||
|
final String url = "/dlqMessage/batchExportDlqMessage.do";
|
||||||
|
{
|
||||||
|
when(mqAdminExt.viewMessage("%DLQ%group_test", "0A9A003F00002A9F0000000000000310"))
|
||||||
|
.thenThrow(new RuntimeException());
|
||||||
|
when(mqAdminExt.viewMessage("%DLQ%group_test", "0A9A003F00002A9F0000000000000311"))
|
||||||
|
.thenReturn(MockObjectUtil.createMessageExt());
|
||||||
|
}
|
||||||
|
List<DlqMessageRequest> dlqMessages = MockObjectUtil.createDlqMessageRequest();
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.contentType(MediaType.APPLICATION_JSON_UTF8);
|
||||||
|
requestBuilder.content(JSON.toJSONString(dlqMessages));
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().is(200))
|
||||||
|
.andExpect(content().contentType("application/vnd.ms-excel;charset=utf-8"));
|
||||||
|
}
|
||||||
|
|
||||||
@Override protected Object getTestController() {
|
@Override protected Object getTestController() {
|
||||||
return dlqMessageController;
|
return dlqMessageController;
|
||||||
}
|
}
|
||||||
|
@@ -62,6 +62,7 @@ public class MessageTraceControllerTest extends BaseControllerTest {
|
|||||||
messageList.add(messageExt);
|
messageList.add(messageExt);
|
||||||
QueryResult queryResult = new QueryResult(System.currentTimeMillis(), messageList);
|
QueryResult queryResult = new QueryResult(System.currentTimeMillis(), messageList);
|
||||||
when(mqAdminExt.queryMessage(anyString(), anyString(), anyInt(), anyLong(), anyLong()))
|
when(mqAdminExt.queryMessage(anyString(), anyString(), anyInt(), anyLong(), anyLong()))
|
||||||
|
.thenThrow(new RuntimeException())
|
||||||
.thenReturn(queryResult);
|
.thenReturn(queryResult);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,6 +101,11 @@ public class MessageTraceControllerTest extends BaseControllerTest {
|
|||||||
final String url = "/messageTrace/viewMessageTraceDetail.query";
|
final String url = "/messageTrace/viewMessageTraceDetail.query";
|
||||||
requestBuilder = MockMvcRequestBuilders.get(url);
|
requestBuilder = MockMvcRequestBuilders.get(url);
|
||||||
requestBuilder.param("msgId", "0A9A003F00002A9F0000000000000319");
|
requestBuilder.param("msgId", "0A9A003F00002A9F0000000000000319");
|
||||||
|
// query message trace exception
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
performErrorExpect(perform);
|
||||||
|
|
||||||
|
// query message trace success
|
||||||
perform = mockMvc.perform(requestBuilder);
|
perform = mockMvc.perform(requestBuilder);
|
||||||
perform.andExpect(status().isOk())
|
perform.andExpect(status().isOk())
|
||||||
.andExpect(jsonPath("$.data", hasSize(4)))
|
.andExpect(jsonPath("$.data", hasSize(4)))
|
||||||
@@ -114,6 +120,11 @@ public class MessageTraceControllerTest extends BaseControllerTest {
|
|||||||
final String url = "/messageTrace/viewMessageTraceGraph.query";
|
final String url = "/messageTrace/viewMessageTraceGraph.query";
|
||||||
requestBuilder = MockMvcRequestBuilders.get(url);
|
requestBuilder = MockMvcRequestBuilders.get(url);
|
||||||
requestBuilder.param("msgId", "0A9A003F00002A9F0000000000000319");
|
requestBuilder.param("msgId", "0A9A003F00002A9F0000000000000319");
|
||||||
|
// query message trace exception
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
performErrorExpect(perform);
|
||||||
|
|
||||||
|
// query message trace success
|
||||||
perform = mockMvc.perform(requestBuilder);
|
perform = mockMvc.perform(requestBuilder);
|
||||||
perform.andExpect(status().isOk())
|
perform.andExpect(status().isOk())
|
||||||
.andExpect(jsonPath("$.data").isMap())
|
.andExpect(jsonPath("$.data").isMap())
|
||||||
|
@@ -34,7 +34,7 @@ public class NamesvrControllerTest extends BaseControllerTest {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testNsaddr() throws Exception {
|
public void testNsaddr() throws Exception {
|
||||||
final String url = "/rocketmq/nsaddr";
|
final String url = "/rocketmq/nsaddr.query";
|
||||||
{
|
{
|
||||||
super.mockRmqConfigure();
|
super.mockRmqConfigure();
|
||||||
}
|
}
|
||||||
|
@@ -35,6 +35,7 @@ import org.springframework.test.util.ReflectionTestUtils;
|
|||||||
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders;
|
import org.springframework.test.web.servlet.request.MockMvcRequestBuilders;
|
||||||
|
|
||||||
import static org.hamcrest.Matchers.hasSize;
|
import static org.hamcrest.Matchers.hasSize;
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
import static org.mockito.ArgumentMatchers.anyString;
|
import static org.mockito.ArgumentMatchers.anyString;
|
||||||
import static org.mockito.Mockito.doNothing;
|
import static org.mockito.Mockito.doNothing;
|
||||||
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
|
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
|
||||||
@@ -64,7 +65,7 @@ public class OpsControllerTest extends BaseControllerTest {
|
|||||||
.andExpect(jsonPath("$.data").isMap())
|
.andExpect(jsonPath("$.data").isMap())
|
||||||
.andExpect(jsonPath("$.data.useVIPChannel").value(false))
|
.andExpect(jsonPath("$.data.useVIPChannel").value(false))
|
||||||
.andExpect(jsonPath("$.data.namesvrAddrList").isArray())
|
.andExpect(jsonPath("$.data.namesvrAddrList").isArray())
|
||||||
.andExpect(jsonPath("$.data.namesvrAddrList", hasSize(1)))
|
.andExpect(jsonPath("$.data.namesvrAddrList", hasSize(2)))
|
||||||
.andExpect(jsonPath("$.data.namesvrAddrList[0]").value("127.0.0.1:9876"));
|
.andExpect(jsonPath("$.data.namesvrAddrList[0]").value("127.0.0.1:9876"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -83,6 +84,20 @@ public class OpsControllerTest extends BaseControllerTest {
|
|||||||
Assert.assertEquals(configure.getNamesrvAddr(), "127.0.0.1:9876");
|
Assert.assertEquals(configure.getNamesrvAddr(), "127.0.0.1:9876");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAddNameSvrAddr() throws Exception {
|
||||||
|
final String url = "/ops/addNameSvrAddr.do";
|
||||||
|
{
|
||||||
|
doNothing().when(configure).setNamesrvAddrs(any());
|
||||||
|
}
|
||||||
|
requestBuilder = MockMvcRequestBuilders.post(url);
|
||||||
|
requestBuilder.param("newNamesrvAddr", "127.0.0.3:9876");
|
||||||
|
perform = mockMvc.perform(requestBuilder);
|
||||||
|
perform.andExpect(status().isOk())
|
||||||
|
.andExpect(jsonPath("$.data").value(true));
|
||||||
|
Assert.assertEquals(configure.getNamesrvAddrs().size(), 3);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testUpdateIsVIPChannel() throws Exception {
|
public void testUpdateIsVIPChannel() throws Exception {
|
||||||
final String url = "/ops/updateIsVIPChannel.do";
|
final String url = "/ops/updateIsVIPChannel.do";
|
||||||
@@ -96,7 +111,6 @@ public class OpsControllerTest extends BaseControllerTest {
|
|||||||
.andExpect(jsonPath("$.data").value(true));
|
.andExpect(jsonPath("$.data").value(true));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testUpdateUseTLS() throws Exception {
|
public void testUpdateUseTLS() throws Exception {
|
||||||
final String url = "/ops/updateUseTLS.do";
|
final String url = "/ops/updateUseTLS.do";
|
||||||
|
@@ -30,6 +30,8 @@ import java.util.HashSet;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
import org.apache.rocketmq.common.MixAll;
|
import org.apache.rocketmq.common.MixAll;
|
||||||
import org.apache.rocketmq.common.protocol.body.BrokerStatsData;
|
import org.apache.rocketmq.common.protocol.body.BrokerStatsData;
|
||||||
import org.apache.rocketmq.common.protocol.body.ClusterInfo;
|
import org.apache.rocketmq.common.protocol.body.ClusterInfo;
|
||||||
@@ -38,6 +40,7 @@ import org.apache.rocketmq.common.protocol.body.KVTable;
|
|||||||
import org.apache.rocketmq.common.protocol.body.TopicList;
|
import org.apache.rocketmq.common.protocol.body.TopicList;
|
||||||
import org.apache.rocketmq.common.protocol.route.TopicRouteData;
|
import org.apache.rocketmq.common.protocol.route.TopicRouteData;
|
||||||
import org.apache.rocketmq.dashboard.BaseTest;
|
import org.apache.rocketmq.dashboard.BaseTest;
|
||||||
|
import org.apache.rocketmq.dashboard.config.CollectExecutorConfig;
|
||||||
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
import org.apache.rocketmq.dashboard.config.RMQConfigure;
|
||||||
import org.apache.rocketmq.dashboard.service.impl.DashboardCollectServiceImpl;
|
import org.apache.rocketmq.dashboard.service.impl.DashboardCollectServiceImpl;
|
||||||
import org.apache.rocketmq.dashboard.util.JsonUtil;
|
import org.apache.rocketmq.dashboard.util.JsonUtil;
|
||||||
@@ -68,6 +71,9 @@ public class DashboardCollectTaskTest extends BaseTest {
|
|||||||
@Mock
|
@Mock
|
||||||
private RMQConfigure rmqConfigure;
|
private RMQConfigure rmqConfigure;
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private ExecutorService collectExecutor;
|
||||||
|
|
||||||
private int taskExecuteNum = 10;
|
private int taskExecuteNum = 10;
|
||||||
|
|
||||||
private File brokerFile;
|
private File brokerFile;
|
||||||
@@ -96,6 +102,7 @@ public class DashboardCollectTaskTest extends BaseTest {
|
|||||||
{
|
{
|
||||||
TopicList topicList = new TopicList();
|
TopicList topicList = new TopicList();
|
||||||
Set<String> topicSet = new HashSet<>();
|
Set<String> topicSet = new HashSet<>();
|
||||||
|
topicSet.add("rmq_sys_xxx");
|
||||||
topicSet.add("topic_test");
|
topicSet.add("topic_test");
|
||||||
topicSet.add("%RETRY%group_test");
|
topicSet.add("%RETRY%group_test");
|
||||||
topicSet.add("%DLQ%group_test");
|
topicSet.add("%DLQ%group_test");
|
||||||
@@ -121,19 +128,35 @@ public class DashboardCollectTaskTest extends BaseTest {
|
|||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
Assert.assertEquals(e.getMessage(), "fetchAllTopicList exception");
|
Assert.assertEquals(e.getMessage(), "fetchAllTopicList exception");
|
||||||
}
|
}
|
||||||
|
dashboardCollectTask.collectTopic();
|
||||||
|
|
||||||
|
// multiple topic collection
|
||||||
|
CollectExecutorConfig config = new CollectExecutorConfig();
|
||||||
|
config.setCoreSize(10);
|
||||||
|
config.setMaxSize(10);
|
||||||
|
config.setQueueSize(500);
|
||||||
|
config.setKeepAliveTime(3000);
|
||||||
|
ExecutorService collectExecutor = config.collectExecutor(config);
|
||||||
for (int i = 0; i < taskExecuteNum; i++) {
|
for (int i = 0; i < taskExecuteNum; i++) {
|
||||||
dashboardCollectTask.collectTopic();
|
CollectTaskRunnble collectTask = new CollectTaskRunnble("topic_test" + i, mqAdminExt, dashboardCollectService);
|
||||||
|
collectExecutor.submit(collectTask);
|
||||||
}
|
}
|
||||||
|
collectExecutor.shutdown();
|
||||||
|
boolean loop = true;
|
||||||
|
do {
|
||||||
|
// Wait for all collectTasks to complete
|
||||||
|
loop = !collectExecutor.awaitTermination(Long.MAX_VALUE, TimeUnit.MINUTES);
|
||||||
|
}
|
||||||
|
while (loop);
|
||||||
LoadingCache<String, List<String>> map = dashboardCollectService.getTopicMap();
|
LoadingCache<String, List<String>> map = dashboardCollectService.getTopicMap();
|
||||||
Assert.assertEquals(map.size(), 1);
|
Assert.assertEquals(map.size(), taskExecuteNum);
|
||||||
Assert.assertEquals(map.get("topic_test").size(), taskExecuteNum);
|
|
||||||
dashboardCollectTask.saveData();
|
dashboardCollectTask.saveData();
|
||||||
Assert.assertEquals(topicFile.exists(), true);
|
Assert.assertEquals(topicFile.exists(), true);
|
||||||
Map<String, List<String>> topicData =
|
Map<String, List<String>> topicData =
|
||||||
JsonUtil.string2Obj(MixAll.file2String(topicFile),
|
JsonUtil.string2Obj(MixAll.file2String(topicFile),
|
||||||
new TypeReference<Map<String, List<String>>>() {
|
new TypeReference<Map<String, List<String>>>() {
|
||||||
});
|
});
|
||||||
Assert.assertEquals(topicData.get("topic_test").size(), taskExecuteNum);
|
Assert.assertEquals(topicData.size(), taskExecuteNum);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@@ -187,8 +210,8 @@ public class DashboardCollectTaskTest extends BaseTest {
|
|||||||
|
|
||||||
private void mockBrokerFileExistBeforeSaveData() throws Exception {
|
private void mockBrokerFileExistBeforeSaveData() throws Exception {
|
||||||
Map<String, List<String>> map = new HashMap<>();
|
Map<String, List<String>> map = new HashMap<>();
|
||||||
map.put("broker-a" + ":" + MixAll.MASTER_ID, Lists.asList("1000", new String[] {"1000"}));
|
map.put("broker-a" + ":" + MixAll.MASTER_ID, Lists.asList("1000", new String[] {"1000"}));
|
||||||
map.put("broker-b" + ":" + MixAll.MASTER_ID, Lists.asList("1000", new String[] {"1000"}));
|
map.put("broker-b" + ":" + MixAll.MASTER_ID, Lists.asList("1000", new String[] {"1000"}));
|
||||||
MixAll.string2File(JsonUtil.obj2String(map), brokerFile.getAbsolutePath());
|
MixAll.string2File(JsonUtil.obj2String(map), brokerFile.getAbsolutePath());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -16,8 +16,10 @@
|
|||||||
*/
|
*/
|
||||||
package org.apache.rocketmq.dashboard.util;
|
package org.apache.rocketmq.dashboard.util;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Arrays;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@@ -30,8 +32,10 @@ import java.util.concurrent.ConcurrentMap;
|
|||||||
import org.apache.rocketmq.client.producer.LocalTransactionState;
|
import org.apache.rocketmq.client.producer.LocalTransactionState;
|
||||||
import org.apache.rocketmq.client.trace.TraceConstants;
|
import org.apache.rocketmq.client.trace.TraceConstants;
|
||||||
import org.apache.rocketmq.client.trace.TraceType;
|
import org.apache.rocketmq.client.trace.TraceType;
|
||||||
|
import org.apache.rocketmq.common.AclConfig;
|
||||||
import org.apache.rocketmq.common.DataVersion;
|
import org.apache.rocketmq.common.DataVersion;
|
||||||
import org.apache.rocketmq.common.MixAll;
|
import org.apache.rocketmq.common.MixAll;
|
||||||
|
import org.apache.rocketmq.common.PlainAccessConfig;
|
||||||
import org.apache.rocketmq.common.TopicConfig;
|
import org.apache.rocketmq.common.TopicConfig;
|
||||||
import org.apache.rocketmq.common.admin.ConsumeStats;
|
import org.apache.rocketmq.common.admin.ConsumeStats;
|
||||||
import org.apache.rocketmq.common.admin.OffsetWrapper;
|
import org.apache.rocketmq.common.admin.OffsetWrapper;
|
||||||
@@ -57,7 +61,9 @@ import org.apache.rocketmq.common.protocol.route.BrokerData;
|
|||||||
import org.apache.rocketmq.common.protocol.route.QueueData;
|
import org.apache.rocketmq.common.protocol.route.QueueData;
|
||||||
import org.apache.rocketmq.common.protocol.route.TopicRouteData;
|
import org.apache.rocketmq.common.protocol.route.TopicRouteData;
|
||||||
import org.apache.rocketmq.common.subscription.SubscriptionGroupConfig;
|
import org.apache.rocketmq.common.subscription.SubscriptionGroupConfig;
|
||||||
|
import org.apache.rocketmq.dashboard.model.DlqMessageRequest;
|
||||||
import org.apache.rocketmq.remoting.protocol.LanguageCode;
|
import org.apache.rocketmq.remoting.protocol.LanguageCode;
|
||||||
|
import org.checkerframework.checker.units.qual.A;
|
||||||
|
|
||||||
import static org.apache.rocketmq.common.protocol.heartbeat.ConsumeType.CONSUME_ACTIVELY;
|
import static org.apache.rocketmq.common.protocol.heartbeat.ConsumeType.CONSUME_ACTIVELY;
|
||||||
|
|
||||||
@@ -298,4 +304,38 @@ public class MockObjectUtil {
|
|||||||
brokerStatsData.setStatsMinute(statsMinute);
|
brokerStatsData.setStatsMinute(statsMinute);
|
||||||
return brokerStatsData;
|
return brokerStatsData;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static List<DlqMessageRequest> createDlqMessageRequest() {
|
||||||
|
List<DlqMessageRequest> dlqMessages = new ArrayList<>();
|
||||||
|
for (int i = 0; i < 2; i++) {
|
||||||
|
DlqMessageRequest dlqMessageRequest = new DlqMessageRequest();
|
||||||
|
dlqMessageRequest.setConsumerGroup("group_test");
|
||||||
|
dlqMessageRequest.setTopicName("topic_test");
|
||||||
|
dlqMessageRequest.setMsgId("0A9A003F00002A9F000000000000031" + i);
|
||||||
|
dlqMessages.add(dlqMessageRequest);
|
||||||
|
}
|
||||||
|
return dlqMessages;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static AclConfig createAclConfig() {
|
||||||
|
PlainAccessConfig adminConfig = new PlainAccessConfig();
|
||||||
|
adminConfig.setAdmin(true);
|
||||||
|
adminConfig.setAccessKey("rocketmq2");
|
||||||
|
adminConfig.setSecretKey("12345678");
|
||||||
|
|
||||||
|
PlainAccessConfig normalConfig = new PlainAccessConfig();
|
||||||
|
normalConfig.setAdmin(false);
|
||||||
|
normalConfig.setAccessKey("rocketmq");
|
||||||
|
normalConfig.setSecretKey("123456789");
|
||||||
|
normalConfig.setDefaultGroupPerm("SUB");
|
||||||
|
normalConfig.setDefaultTopicPerm("DENY");
|
||||||
|
normalConfig.setTopicPerms(Lists.newArrayList("topicA=DENY", "topicB=PUB|SUB"));
|
||||||
|
normalConfig.setGroupPerms(Lists.newArrayList("groupA=DENY", "groupB=PUB|SUB"));
|
||||||
|
|
||||||
|
|
||||||
|
AclConfig aclConfig = new AclConfig();
|
||||||
|
aclConfig.setPlainAccessConfigs(Lists.newArrayList(adminConfig, normalConfig));
|
||||||
|
aclConfig.setGlobalWhiteAddrs(Lists.newArrayList("localhost"));
|
||||||
|
return aclConfig;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user