验证功能是否正常 (#1193)

This commit is contained in:
EricZeng
2023-11-27 13:51:21 +08:00
committed by GitHub
17 changed files with 234 additions and 155 deletions

View File

@@ -146,7 +146,7 @@ PS: 提问请尽量把问题一次性描述清楚,并告知环境信息情况
**`2、微信群`** **`2、微信群`**
微信加群:添加`mike_zhangliang``PenceXie``szzdzhp001`的微信号备注KnowStreaming加群。 微信加群:添加`PenceXie``szzdzhp001`的微信号备注KnowStreaming加群。
<br/> <br/>
加群之前有劳点一下 star一个小小的 star 是对KnowStreaming作者们努力建设社区的动力。 加群之前有劳点一下 star一个小小的 star 是对KnowStreaming作者们努力建设社区的动力。

View File

@@ -6,72 +6,72 @@
### 3.3.1、Cluster 指标 ### 3.3.1、Cluster 指标
| 指标名称 | 指标单位 | 指标含义 | kafka 版本 | 企业/开源版指标 | | 指标名称 | 指标单位 | 指标含义 | kafka 版本 | 企业/开源版指标 |
| ------------------------- | -------- | ------------------------------------ | ---------------- | --------------- | | ------------------------- | -------- |--------------------------------| ---------------- | --------------- |
| HealthScore | 分 | 集群总体的健康分 | 全部版本 | 开源版 | | HealthScore | 分 | 集群总体的健康分 | 全部版本 | 开源版 |
| HealthCheckPassed | 个 | 集群总体健康检查通过数 | 全部版本 | 开源版 | | HealthCheckPassed | 个 | 集群总体健康检查通过数 | 全部版本 | 开源版 |
| HealthCheckTotal | 个 | 集群总体健康检查总数 | 全部版本 | 开源版 | | HealthCheckTotal | 个 | 集群总体健康检查总数 | 全部版本 | 开源版 |
| HealthScore_Topics | 分 | 集群 Topics 的健康分 | 全部版本 | 开源版 | | HealthScore_Topics | 分 | 集群 Topics 的健康分 | 全部版本 | 开源版 |
| HealthCheckPassed_Topics | 个 | 集群 Topics 健康检查通过数 | 全部版本 | 开源版 | | HealthCheckPassed_Topics | 个 | 集群 Topics 健康检查通过数 | 全部版本 | 开源版 |
| HealthCheckTotal_Topics | 个 | 集群 Topics 健康检查总数 | 全部版本 | 开源版 | | HealthCheckTotal_Topics | 个 | 集群 Topics 健康检查总数 | 全部版本 | 开源版 |
| HealthScore_Brokers | 分 | 集群 Brokers 的健康分 | 全部版本 | 开源版 | | HealthScore_Brokers | 分 | 集群 Brokers 的健康分 | 全部版本 | 开源版 |
| HealthCheckPassed_Brokers | 个 | 集群 Brokers 健康检查通过数 | 全部版本 | 开源版 | | HealthCheckPassed_Brokers | 个 | 集群 Brokers 健康检查通过数 | 全部版本 | 开源版 |
| HealthCheckTotal_Brokers | 个 | 集群 Brokers 健康检查总数 | 全部版本 | 开源版 | | HealthCheckTotal_Brokers | 个 | 集群 Brokers 健康检查总数 | 全部版本 | 开源版 |
| HealthScore_Groups | 分 | 集群 Groups 的健康分 | 全部版本 | 开源版 | | HealthScore_Groups | 分 | 集群 Groups 的健康分 | 全部版本 | 开源版 |
| HealthCheckPassed_Groups | 个 | 集群 Groups 健康检查总数 | 全部版本 | 开源版 | | HealthCheckPassed_Groups | 个 | 集群 Groups 健康检查总数 | 全部版本 | 开源版 |
| HealthCheckTotal_Groups | 个 | 集群 Groups 健康检查总数 | 全部版本 | 开源版 | | HealthCheckTotal_Groups | 个 | 集群 Groups 健康检查总数 | 全部版本 | 开源版 |
| HealthScore_Cluster | 分 | 集群自身的健康分 | 全部版本 | 开源版 | | HealthScore_Cluster | 分 | 集群自身的健康分 | 全部版本 | 开源版 |
| HealthCheckPassed_Cluster | 个 | 集群自身健康检查通过数 | 全部版本 | 开源版 | | HealthCheckPassed_Cluster | 个 | 集群自身健康检查通过数 | 全部版本 | 开源版 |
| HealthCheckTotal_Cluster | 个 | 集群自身健康检查总数 | 全部版本 | 开源版 | | HealthCheckTotal_Cluster | 个 | 集群自身健康检查总数 | 全部版本 | 开源版 |
| TotalRequestQueueSize | 个 | 集群中总的请求队列数 | 全部版本 | 开源版 | | TotalRequestQueueSize | 个 | 集群中总的请求队列数 | 全部版本 | 开源版 |
| TotalResponseQueueSize | 个 | 集群中总的响应队列数 | 全部版本 | 开源版 | | TotalResponseQueueSize | 个 | 集群中总的响应队列数 | 全部版本 | 开源版 |
| EventQueueSize | 个 | 集群中 Controller 的 EventQueue 大小 | 2.0.0 及以上版本 | 开源版 | | EventQueueSize | 个 | 集群中 Controller 的 EventQueue 大小 | 2.0.0 及以上版本 | 开源版 |
| ActiveControllerCount | 个 | 集群中存活的 Controller 数 | 全部版本 | 开源版 | | ActiveControllerCount | 个 | 集群中存活的 Controller 数 | 全部版本 | 开源版 |
| TotalProduceRequests | 个 | 集群中的 Produce 每秒请求数 | 全部版本 | 开源版 | | TotalProduceRequests | 个 | 集群中的 Produce 每秒请求数 | 全部版本 | 开源版 |
| TotalLogSize | byte | 集群总的已使用的磁盘大小 | 全部版本 | 开源版 | | TotalLogSize | byte | 集群总的已使用的磁盘大小 | 全部版本 | 开源版 |
| ConnectionsCount | 个 | 集群的连接(Connections)个数 | 全部版本 | 开源版 | | ConnectionsCount | 个 | 集群的连接(Connections)个数 | 全部版本 | 开源版 |
| Zookeepers | 个 | 集群中存活的 zk 节点个数 | 全部版本 | 开源版 | | Zookeepers | 个 | 集群中存活的 zk 节点个数 | 全部版本 | 开源版 |
| ZookeepersAvailable | 是/否 | ZK 地址是否合法 | 全部版本 | 开源版 | | ZookeepersAvailable | 是/否 | ZK 地址是否合法 | 全部版本 | 开源版 |
| Brokers | 个 | 集群的 broker 的总数 | 全部版本 | 开源版 | | Brokers | 个 | 集群的 broker 的总数 | 全部版本 | 开源版 |
| BrokersAlive | 个 | 集群的 broker 的存活数 | 全部版本 | 开源版 | | BrokersAlive | 个 | 集群的 broker 的存活数 | 全部版本 | 开源版 |
| BrokersNotAlive | 个 | 集群的 broker 的未存活数 | 全部版本 | 开源版 | | BrokersNotAlive | 个 | 集群的 broker 的未存活数 | 全部版本 | 开源版 |
| Replicas | 个 | 集群中 Replica 的总数 | 全部版本 | 开源版 | | Replicas | 个 | 集群中 Replica 的总数 | 全部版本 | 开源版 |
| Topics | 个 | 集群中 Topic 的总数 | 全部版本 | 开源版 | | Topics | 个 | 集群中 Topic 的总数 | 全部版本 | 开源版 |
| Partitions | 个 | 集群的 Partitions 总数 | 全部版本 | 开源版 | | Partitions | 个 | 集群的 Partitions 总数 | 全部版本 | 开源版 |
| PartitionNoLeader | 个 | 集群中的 PartitionNoLeader 总数 | 全部版本 | 开源版 | | PartitionNoLeader | 个 | 集群中的 PartitionNoLeader 总数 | 全部版本 | 开源版 |
| PartitionMinISR_S | 个 | 集群中的小于 PartitionMinISR 总数 | 全部版本 | 开源版 | | PartitionMinISR_S | 个 | 集群中的小于 PartitionMinISR 总数 | 全部版本 | 开源版 |
| PartitionMinISR_E | 个 | 集群中的等于 PartitionMinISR 总数 | 全部版本 | 开源版 | | PartitionMinISR_E | 个 | 集群中的等于 PartitionMinISR 总数 | 全部版本 | 开源版 |
| PartitionURP | 个 | 集群中的未同步的 Partition 总数 | 全部版本 | 开源版 | | PartitionURP | 个 | 集群中的未同步的 Partition 总数 | 全部版本 | 开源版 |
| MessagesIn | 条/s | 集群每消息写入条数 | 全部版本 | 开源版 | | MessagesIn | 条/s | 集群每消息写入条数 | 全部版本 | 开源版 |
| Messages | 条 | 集群总的消息条数 | 全部版本 | 开源版 | | Messages | 条 | 集群总的消息条数 | 全部版本 | 开源版 |
| LeaderMessages | 条 | 集群中 leader 总的消息条数 | 全部版本 | 开源版 | | LeaderMessages | 条 | 集群中 leader 总的消息条数 | 全部版本 | 开源版 |
| BytesIn | byte/s | 集群的每秒写入字节数 | 全部版本 | 开源版 | | BytesIn | byte/s | 集群的每秒写入字节数 | 全部版本 | 开源版 |
| BytesIn_min_5 | byte/s | 集群的每秒写入字节数5 分钟均值 | 全部版本 | 开源版 | | BytesIn_min_5 | byte/s | 集群的每秒写入字节数5 分钟均值 | 全部版本 | 开源版 |
| BytesIn_min_15 | byte/s | 集群的每秒写入字节数15 分钟均值 | 全部版本 | 开源版 | | BytesIn_min_15 | byte/s | 集群的每秒写入字节数15 分钟均值 | 全部版本 | 开源版 |
| BytesOut | byte/s | 集群的每秒流出字节数 | 全部版本 | 开源版 | | BytesOut | byte/s | 集群的每秒流出字节数 | 全部版本 | 开源版 |
| BytesOut_min_5 | byte/s | 集群的每秒流出字节数5 分钟均值 | 全部版本 | 开源版 | | BytesOut_min_5 | byte/s | 集群的每秒流出字节数5 分钟均值 | 全部版本 | 开源版 |
| BytesOut_min_15 | byte/s | 集群的每秒流出字节数15 分钟均值 | 全部版本 | 开源版 | | BytesOut_min_15 | byte/s | 集群的每秒流出字节数15 分钟均值 | 全部版本 | 开源版 |
| Groups | 个 | 集群中 Group 的总数 | 全部版本 | 开源版 | | Groups | 个 | 集群中 Group 的总数 | 全部版本 | 开源版 |
| GroupActives | 个 | 集群中 ActiveGroup 的总数 | 全部版本 | 开源版 | | GroupActives | 个 | 集群中 ActiveGroup 的总数 | 全部版本 | 开源版 |
| GroupEmptys | 个 | 集群中 EmptyGroup 的总数 | 全部版本 | 开源版 | | GroupEmptys | 个 | 集群中 EmptyGroup 的总数 | 全部版本 | 开源版 |
| GroupRebalances | 个 | 集群中 RebalanceGroup 的总数 | 全部版本 | 开源版 | | GroupRebalances | 个 | 集群中 RebalanceGroup 的总数 | 全部版本 | 开源版 |
| GroupDeads | 个 | 集群中 DeadGroup 的总数 | 全部版本 | 开源版 | | GroupDeads | 个 | 集群中 DeadGroup 的总数 | 全部版本 | 开源版 |
| Alive | 是/否 | 集群是否存活1存活0没有存活 | 全部版本 | 开源版 | | Alive | 是/否 | 集群是否存活1存活0没有存活 | 全部版本 | 开源版 |
| AclEnable | 是/否 | 集群是否开启 Acl10否 | 全部版本 | 开源版 | | AclEnable | 是/否 | 集群是否开启 Acl10 | 全部版本 | 开源版 |
| Acls | 个 | ACL 数 | 全部版本 | 开源版 | | Acls | 个 | ACL 数 | 全部版本 | 开源版 |
| AclUsers | 个 | ACL-KafkaUser 数 | 全部版本 | 开源版 | | AclUsers | 个 | ACL-KafkaUser 数 | 全部版本 | 开源版 |
| AclTopics | 个 | ACL-Topic 数 | 全部版本 | 开源版 | | AclTopics | 个 | ACL-Topic 数 | 全部版本 | 开源版 |
| AclGroups | 个 | ACL-Group 数 | 全部版本 | 开源版 | | AclGroups | 个 | ACL-Group 数 | 全部版本 | 开源版 |
| Jobs | 个 | 集群任务总数 | 全部版本 | 开源版 | | Jobs | 个 | 集群任务总数 | 全部版本 | 开源版 |
| JobsRunning | 个 | 集群 running 任务总数 | 全部版本 | 开源版 | | JobsRunning | 个 | 集群 running 任务总数 | 全部版本 | 开源版 |
| JobsWaiting | 个 | 集群 waiting 任务总数 | 全部版本 | 开源版 | | JobsWaiting | 个 | 集群 waiting 任务总数 | 全部版本 | 开源版 |
| JobsSuccess | 个 | 集群 success 任务总数 | 全部版本 | 开源版 | | JobsSuccess | 个 | 集群 success 任务总数 | 全部版本 | 开源版 |
| JobsFailed | 个 | 集群 failed 任务总数 | 全部版本 | 开源版 | | JobsFailed | 个 | 集群 failed 任务总数 | 全部版本 | 开源版 |
| LoadReBalanceEnable | 是/否 | 是否开启均衡, 10否 | 全部版本 | 企业版 | | LoadReBalanceEnable | 是/否 | 是否开启均衡, 10 | 全部版本 | 企业版 |
| LoadReBalanceCpu | 是/否 | CPU 是否均衡, 10否 | 全部版本 | 企业版 | | LoadReBalanceCpu | 是/否 | CPU 是否均衡, 10 | 全部版本 | 企业版 |
| LoadReBalanceNwIn | 是/否 | BytesIn 是否均衡, 10否 | 全部版本 | 企业版 | | LoadReBalanceNwIn | 是/否 | BytesIn 是否均衡, 10 | 全部版本 | 企业版 |
| LoadReBalanceNwOut | 是/否 | BytesOut 是否均衡, 10否 | 全部版本 | 企业版 | | LoadReBalanceNwOut | 是/否 | BytesOut 是否均衡, 10 | 全部版本 | 企业版 |
| LoadReBalanceDisk | 是/否 | Disk 是否均衡, 10否 | 全部版本 | 企业版 | | LoadReBalanceDisk | 是/否 | Disk 是否均衡, 10 | 全部版本 | 企业版 |
### 3.3.2、Broker 指标 ### 3.3.2、Broker 指标

View File

@@ -22,6 +22,7 @@
- [16、JMX连接失败怎么办](#16jmx连接失败怎么办) - [16、JMX连接失败怎么办](#16jmx连接失败怎么办)
- [17、zk监控无数据问题](#17zk监控无数据问题) - [17、zk监控无数据问题](#17zk监控无数据问题)
- [18、启动失败报NoClassDefFoundError如何解决](#18启动失败报noclassdeffounderror如何解决) - [18、启动失败报NoClassDefFoundError如何解决](#18启动失败报noclassdeffounderror如何解决)
- [19、依赖ElasticSearch 8.0以上版本部署后指标信息无法正常显示如何解决]
## 1、支持哪些 Kafka 版本? ## 1、支持哪些 Kafka 版本?
@@ -306,4 +307,15 @@ at org.springframework.beans.factory.support.ConstructorResolver.instantiate(Con
如果在在 `Windows``Mac``CentOS` 这几个操作系统下也出现了启动失败的问题可以重试2-3次看是否还是启动失败或者换一台机器试试。 如果在在 `Windows``Mac``CentOS` 这几个操作系统下也出现了启动失败的问题可以重试2-3次看是否还是启动失败或者换一台机器试试。
## 依赖ElasticSearch 8.0以上版本部署后指标信息无法正常显示如何解决
**错误现象**
```log
Warnings: [299 Elasticsearch-8.9.1-a813d015ef1826148d9d389bd1c0d781c6e349f0 "Legacy index templates are deprecated in favor of composable templates."]
```
**问题原因**
1. ES8.0和ES7.0版本存在Template模式的差异建议使用 /_index_template 端点来管理模板;
2. ES java client在此版本的行为很奇怪表现为读取数据为空
**解决方法**
修改`es_template_create.sh`脚本中所有的`/_template``/_index_template`后执行即可。

View File

@@ -16,6 +16,13 @@ const babelOptions = {
cacheDirectory: true, cacheDirectory: true,
babelrc: false, babelrc: false,
presets: [require.resolve('@babel/preset-env'), require.resolve('@babel/preset-typescript'), require.resolve('@babel/preset-react')], presets: [require.resolve('@babel/preset-env'), require.resolve('@babel/preset-typescript'), require.resolve('@babel/preset-react')],
overrides: [
// TODO编译时需要做的事情更多应该只针对目标第三方库
{
include: './node_modules',
sourceType: 'unambiguous'
}
],
plugins: [ plugins: [
[require.resolve('@babel/plugin-proposal-decorators'), { legacy: true }], [require.resolve('@babel/plugin-proposal-decorators'), { legacy: true }],
[require.resolve('@babel/plugin-proposal-class-properties'), { loose: true }], [require.resolve('@babel/plugin-proposal-class-properties'), { loose: true }],

View File

@@ -108,6 +108,7 @@
"optimize-css-assets-webpack-plugin": "^5.0.1", "optimize-css-assets-webpack-plugin": "^5.0.1",
"prettier": "2.3.2", "prettier": "2.3.2",
"progress-bar-webpack-plugin": "^1.12.1", "progress-bar-webpack-plugin": "^1.12.1",
"pubsub-js": "^1.9.4",
"query-string": "^7.0.1", "query-string": "^7.0.1",
"react-refresh": "^0.10.0", "react-refresh": "^0.10.0",
"react-router-dom": "5.2.1", "react-router-dom": "5.2.1",

View File

@@ -49,7 +49,7 @@ const ConnectDetailCard = (props: { record: any }) => {
return ( return (
<> <>
{ {
<span style={{ fontFamily: 'HelveticaNeue-Medium', fontSize: 32, color: '#212529' }}> <span style={{ fontFamily: 'HelveticaNeue-Medium', fontSize: 28, color: '#212529' }}>
{Utils.firstCharUppercase(type) || '-'} {Utils.firstCharUppercase(type) || '-'}
</span> </span>
} }
@@ -64,7 +64,7 @@ const ConnectDetailCard = (props: { record: any }) => {
return ( return (
<> <>
{ {
<span style={{ fontFamily: 'HelveticaNeue-Medium', fontSize: 32, color: stateEnum[state].color }}> <span style={{ fontFamily: 'HelveticaNeue-Medium', fontSize: 28, color: stateEnum[state].color }}>
{Utils.firstCharUppercase(state) || '-'} {Utils.firstCharUppercase(state) || '-'}
</span> </span>
} }

View File

@@ -48,6 +48,12 @@ export enum ClustersPermissionMap {
CONNECTOR_DELETE = 'Connector-删除', CONNECTOR_DELETE = 'Connector-删除',
CONNECTOR_RESTART = 'Connector-重启', CONNECTOR_RESTART = 'Connector-重启',
CONNECTOR_STOP_RESUME = 'Connector-暂停&恢复', CONNECTOR_STOP_RESUME = 'Connector-暂停&恢复',
// Security
SECURITY_ACL_ADD = 'Security-ACL新增',
SECURITY_ACL_DELETE = 'Security-ACL删除',
SECURITY_USER_ADD = 'Security-User新增',
SECURITY_USER_DELETE = 'Security-User删除',
SECURITY_USER_EDIT_PASSWORD = 'Security-User修改密码',
} }
export interface PermissionNode { export interface PermissionNode {
@@ -98,9 +104,7 @@ const CommonConfig = () => {
clustersPermissions.childList.forEach((node: PermissionNode) => node.has && userPermissions.push(node.permissionName)); clustersPermissions.childList.forEach((node: PermissionNode) => node.has && userPermissions.push(node.permissionName));
// 获取用户在系统管理拥有的权限 // 获取用户在系统管理拥有的权限
const configPermissions = userPermissionTree.find( const configPermissions = userPermissionTree.find((sys: PermissionNode) => sys.permissionName === ClustersPermissionMap.SYS_MANAGE);
(sys: PermissionNode) => sys.permissionName === ClustersPermissionMap.SYS_MANAGE
);
configPermissions && configPermissions &&
configPermissions.childList.forEach((node: PermissionNode) => node.has && userPermissions.push(node.permissionName)); configPermissions.childList.forEach((node: PermissionNode) => node.has && userPermissions.push(node.permissionName));

View File

@@ -1,7 +1,7 @@
import api from '@src/api'; import api from '@src/api';
import CodeMirrorFormItem from '@src/components/CodeMirrorFormItem'; import CodeMirrorFormItem from '@src/components/CodeMirrorFormItem';
import customMessage from '@src/components/Message'; import customMessage from '@src/components/Message';
import { Button, Divider, Drawer, Form, message, Space, Utils } from 'knowdesign'; import { Button, Divider, Drawer, Form, message, Space, Utils, Select } from 'knowdesign';
import React, { forwardRef, useEffect, useImperativeHandle, useState } from 'react'; import React, { forwardRef, useEffect, useImperativeHandle, useState } from 'react';
import { useParams } from 'react-router-dom'; import { useParams } from 'react-router-dom';
import { ConnectCluster, ConnectorPlugin, ConnectorPluginConfig, OperateInfo } from './AddConnector'; import { ConnectCluster, ConnectorPlugin, ConnectorPluginConfig, OperateInfo } from './AddConnector';
@@ -9,9 +9,8 @@ import { ConnectCluster, ConnectorPlugin, ConnectorPluginConfig, OperateInfo } f
const PLACEHOLDER = `配置格式如下 const PLACEHOLDER = `配置格式如下
{ {
"connectClusterName": "", // Connect Cluster 名称 "name": "", // Connect Cluster 名称
"config": { // 具体配置项 "config": { // 具体配置项
"name": "",
"connector.class": "", "connector.class": "",
"tasks.max": 1, "tasks.max": 1,
... ...
@@ -43,11 +42,16 @@ export default forwardRef((props: any, ref) => {
const onOpen = (type: 'create' | 'edit', connectClusterName?: string, defaultConfigs?: { [key: string]: any }) => { const onOpen = (type: 'create' | 'edit', connectClusterName?: string, defaultConfigs?: { [key: string]: any }) => {
if (defaultConfigs) { if (defaultConfigs) {
setDefaultConfigs({ ...defaultConfigs, connectClusterName }); setDefaultConfigs({ ...defaultConfigs, connectClusterName });
const connectorName = connectClusterName;
const connectClusterId = connectClusters.find((cluster) => cluster.label === connectClusterName).value;
form.setFieldsValue({ form.setFieldsValue({
connectClusterId,
connectorName,
configs: JSON.stringify( configs: JSON.stringify(
{ {
connectClusterName, // connectClusterName,
config: defaultConfigs, name: defaultConfigs.name,
config: { ...defaultConfigs, name: undefined },
}, },
null, null,
2 2
@@ -63,10 +67,11 @@ export default forwardRef((props: any, ref) => {
form.validateFields().then( form.validateFields().then(
(data) => { (data) => {
const postData = JSON.parse(data.configs); const postData = JSON.parse(data.configs);
postData.connectorName = postData.config.name; postData.connectorName = postData.name;
postData.connectClusterId = connectClusters.find((cluster) => cluster.label === postData.connectClusterName).value; postData.connectClusterId = data.connectClusterId;
delete postData.connectClusterName; postData.config.name = postData.name;
// delete postData.connectClusterName;
delete postData.name;
Object.entries(postData.config).forEach(([key, val]) => { Object.entries(postData.config).forEach(([key, val]) => {
if (val === null) { if (val === null) {
delete postData.config[key]; delete postData.config[key];
@@ -161,6 +166,26 @@ export default forwardRef((props: any, ref) => {
} }
> >
<Form form={form} layout="vertical"> <Form form={form} layout="vertical">
<Form.Item
name="connectClusterId"
label="Connect 集群"
rules={[
{
required: true,
validator(rule, value) {
if (!value) {
return Promise.reject('Connect 集群不能为空');
} else {
return Promise.resolve();
}
},
},
]}
initialValue={defaultConfigs?.connectClusterId}
className="connector-json-connectCluster"
>
<Select options={connectClusters} placeholder="请选择 Connect 集群" disabled={type === 'edit'} />
</Form.Item>
<Form.Item <Form.Item
name="configs" name="configs"
validateTrigger="onBlur" validateTrigger="onBlur"
@@ -175,40 +200,27 @@ export default forwardRef((props: any, ref) => {
if (typeof v !== 'object') { if (typeof v !== 'object') {
return Promise.reject('输入内容必须为 JSON'); return Promise.reject('输入内容必须为 JSON');
} }
let connectClusterId = -1; let connectClusterId = form.getFieldValue('connectClusterId');
// 校验 connectClusterName 字段 // 校验 connectorName 字段
if (!v.connectClusterName) { if (!v.name) {
return Promise.reject('内容缺少 connectClusterName 字段或字段内容为空'); return Promise.reject('内容缺少 name 项');
} else { } else {
if (type === 'edit') { if (type === 'edit' && v.name !== defaultConfigs.name) {
if (v.connectClusterName !== defaultConfigs.connectClusterName) { return Promise.reject('编辑模式下不允许修改 name 字段');
return Promise.reject('编辑模式下不允许修改 connectClusterName 字段');
}
} else {
if (!connectClusters.length) {
getConnectClusters();
return Promise.reject('connectClusterName 列表获取失败,请重试');
}
const targetConnectCluster = connectClusters.find((cluster) => cluster.label === v.connectClusterName);
if (!targetConnectCluster) {
return Promise.reject('connectClusterName 不存在,请检查');
} else {
connectClusterId = targetConnectCluster.value;
}
} }
} }
if (!v.config || typeof v.config !== 'object') { if (!v.config || typeof v.config !== 'object') {
return Promise.reject('内容缺少 config 字段或字段格式错误'); return Promise.reject('内容缺少 config 字段或字段格式错误');
} else { } else {
// 校验 connectorName 字段 // // 校验 connectorName 字段
if (!v.config.name) { // if (!v.config.name) {
return Promise.reject('config 字段下缺少 name 项'); // return Promise.reject('config 字段下缺少 name 项');
} else { // } else {
if (type === 'edit' && v.config.name !== defaultConfigs.name) { // if (type === 'edit' && v.config.name !== defaultConfigs.name) {
return Promise.reject('编辑模式下不允许修改 name 字段'); // return Promise.reject('编辑模式下不允许修改 name 字段');
} // }
} // }
if (!v.config['connector.class']) { if (!v.config['connector.class']) {
return Promise.reject('config 字段下缺少 connector.class 项'); return Promise.reject('config 字段下缺少 connector.class 项');
} else if (type === 'edit' && v.config['connector.class'] !== defaultConfigs['connector.class']) { } else if (type === 'edit' && v.config['connector.class'] !== defaultConfigs['connector.class']) {
@@ -217,6 +229,10 @@ export default forwardRef((props: any, ref) => {
} }
if (type === 'create') { if (type === 'create') {
// 校验创建时是否选择了connect集群
if (!connectClusterId) {
return Promise.reject('请先选择 Connect 集群');
}
// 异步校验 connector 名称是否重复 以及 className 是否存在 // 异步校验 connector 名称是否重复 以及 className 是否存在
return Promise.all([ return Promise.all([
Utils.request(api.isConnectorExist(connectClusterId, v.config.name)), Utils.request(api.isConnectorExist(connectClusterId, v.config.name)),

View File

@@ -315,6 +315,7 @@ export const getWorkersColumns = (arg?: any) => {
// Detail // Detail
export const getConnectorsDetailColumns = (arg?: any) => { export const getConnectorsDetailColumns = (arg?: any) => {
const [global] = AppContainer.useGlobalValue();
const columns = [ const columns = [
{ {
title: 'Task ID', title: 'Task ID',
@@ -363,16 +364,20 @@ export const getConnectorsDetailColumns = (arg?: any) => {
render: (_t: any, r: any) => { render: (_t: any, r: any) => {
return ( return (
<div> <div>
<Popconfirm {global.hasPermission(ClustersPermissionMap.CONNECTOR_RESTART) ? (
title="是否重试当前任务?" <Popconfirm
onConfirm={() => arg?.retryOption(r.taskId)} title="是否重试当前任务?"
// onCancel={cancel} onConfirm={() => arg?.retryOption(r.taskId)}
okText="是" // onCancel={cancel}
cancelText="" okText=""
overlayClassName="connect-popconfirm" cancelText="否"
> overlayClassName="connect-popconfirm"
<a></a> >
</Popconfirm> <a></a>
</Popconfirm>
) : (
<></>
)}
</div> </div>
); );
}, },

View File

@@ -185,9 +185,10 @@
.operate-connector-drawer-use-json { .operate-connector-drawer-use-json {
.CodeMirror.cm-s-default { .CodeMirror.cm-s-default {
height: calc(100vh - 146px); height: calc(100vh - 196px);
} }
.dcloud-form-item { .dcloud-form-item {
margin-top: 16px;
margin-bottom: 0 !important; margin-bottom: 0 !important;
} }
} }

View File

@@ -4,7 +4,7 @@ import { useParams } from 'react-router-dom';
import EditTable from '../TestingProduce/component/EditTable'; import EditTable from '../TestingProduce/component/EditTable';
import Api from '@src/api/index'; import Api from '@src/api/index';
import moment from 'moment'; import moment from 'moment';
import PubSub from 'pubsub-js' import PubSub from 'pubsub-js';
const CustomSelectResetTime = (props: { value?: string; onChange?: (val: Number | String) => void }) => { const CustomSelectResetTime = (props: { value?: string; onChange?: (val: Number | String) => void }) => {
const { value, onChange } = props; const { value, onChange } = props;

View File

@@ -13,7 +13,7 @@ const carouselList = [
<img className="carousel-eg-ctr-two-img img-one" src={egTwoContent} /> <img className="carousel-eg-ctr-two-img img-one" src={egTwoContent} />
<div className="carousel-eg-ctr-two-desc desc-one"> <div className="carousel-eg-ctr-two-desc desc-one">
<span>Github: </span> <span>Github: </span>
<span>5.8K</span> <span>6.8K</span>
<span>+ Star的的实时流处理平台</span> <span>+ Star的的实时流处理平台</span>
</div> </div>
<div className="carousel-eg-ctr-two-desc desc-two"> <div className="carousel-eg-ctr-two-desc desc-two">

View File

@@ -185,7 +185,7 @@
.operate-connector-drawer-use-json { .operate-connector-drawer-use-json {
.CodeMirror.cm-s-default { .CodeMirror.cm-s-default {
height: calc(100vh - 146px); height: calc(100vh - 196px);
} }
.dcloud-form-item { .dcloud-form-item {
margin-bottom: 0 !important; margin-bottom: 0 !important;

View File

@@ -132,17 +132,35 @@ const AddDrawer = forwardRef((_, ref) => {
form.validateFields().then((formData) => { form.validateFields().then((formData) => {
const submitData = []; const submitData = [];
const { configType, principle, kafkaUser } = formData; const { configType, principle, kafkaUser } = formData;
if (configType === 'custom') { if (configType === 'custom') {
// 1. 自定义权限 // 1. 自定义权限
// TODO: 需要和后端联调 // TODO: 需要和后端联调
const { resourceType, resourcePatternType, aclPermissionType, aclOperation, aclClientHost } = formData; const {
resourceType,
resourcePatternType,
aclPermissionType,
aclOperation,
aclClientHost,
cluster,
topicName,
topicPatternType,
groupName,
groupPatternType,
transactionalId,
transactionalIdPatternType,
} = formData;
submitData.push({ submitData.push({
clusterId, clusterId,
kafkaUser: principle === 'all' ? '*' : kafkaUser, kafkaUser: principle === 'all' ? '*' : kafkaUser,
resourceType, resourceType,
resourcePatternType, resourcePatternType: cluster
resourceName: '*', ? 3
: topicPatternType
? topicPatternType
: groupPatternType
? groupPatternType
: transactionalIdPatternType,
resourceName: cluster ? cluster : topicName ? topicName : groupName ? groupName : transactionalId,
aclPermissionType, aclPermissionType,
aclOperation, aclOperation,
aclClientHost, aclClientHost,
@@ -348,37 +366,43 @@ const AddDrawer = forwardRef((_, ref) => {
<Form.Item dependencies={[`${type}Principle`]} style={{ marginBottom: 0 }}> <Form.Item dependencies={[`${type}Principle`]} style={{ marginBottom: 0 }}>
{({ getFieldValue }) => {({ getFieldValue }) =>
getFieldValue(`${type}Principle`) === 'special' ? ( getFieldValue(`${type}Principle`) === 'special' ? (
<Form.Item type !== 'transactionalId' ? (
name={`${type}Name`} <Form.Item
dependencies={[`${type}PatternType`]} name={`${type}Name`}
validateTrigger="onBlur" dependencies={[`${type}PatternType`]}
rules={[ validateTrigger="onBlur"
({ getFieldValue }) => ({ rules={[
validator: (rule: any, value: string) => { ({ getFieldValue }) => ({
if (!value) { validator: (rule: any, value: string) => {
return Promise.reject(`${UpperCaseType}Name 不能为空`); if (!value) {
return Promise.reject(`${UpperCaseType}Name 不能为空`);
}
if (type === 'topic' && getFieldValue(`${type}PatternType`) === ACL_PATTERN_TYPE['Literal']) {
return Utils.request(api.getTopicMetadata(clusterId as any, value)).then((res: any) => {
return res?.exist ? Promise.resolve() : Promise.reject('该 Topic 不存在');
});
}
return Promise.resolve();
},
}),
]}
>
<AutoComplete
filterOption={(value, option) => {
if (option?.value.includes(value)) {
return true;
} }
if (type === 'topic' && getFieldValue(`${type}PatternType`) === ACL_PATTERN_TYPE['Literal']) { return false;
return Utils.request(api.getTopicMetadata(clusterId as any, value)).then((res: any) => { }}
return res?.exist ? Promise.resolve() : Promise.reject('该 Topic 不存在'); options={type === 'topic' ? topicMetaData : groupMetaData}
}); placeholder={`请输入 ${type}Name`}
} />
return Promise.resolve(); </Form.Item>
}, ) : (
}), <Form.Item name={`transactionalId`} rules={[{ required: true, message: `TransactionalId不能为空` }]}>
]} <Input placeholder={`请输入TransactionalId`}></Input>
> </Form.Item>
<AutoComplete )
filterOption={(value, option) => {
if (option?.value.includes(value)) {
return true;
}
return false;
}}
options={type === 'topic' ? topicMetaData : groupMetaData}
placeholder={`请输入 ${type}Name`}
/>
</Form.Item>
) : null ) : null
} }
</Form.Item> </Form.Item>
@@ -400,7 +424,7 @@ const AddDrawer = forwardRef((_, ref) => {
<Radio value={ACL_PERMISSION_TYPE['Deny']}>Deny</Radio> <Radio value={ACL_PERMISSION_TYPE['Deny']}>Deny</Radio>
</Radio.Group> </Radio.Group>
</Form.Item> </Form.Item>
<Form.Item {/* <Form.Item
label="Pattern Type" label="Pattern Type"
name="resourcePatternType" name="resourcePatternType"
rules={[{ required: true, message: 'Pattern Type 不能为空' }]} rules={[{ required: true, message: 'Pattern Type 不能为空' }]}
@@ -410,7 +434,7 @@ const AddDrawer = forwardRef((_, ref) => {
<Radio value={ACL_PATTERN_TYPE['Literal']}>Literal</Radio> <Radio value={ACL_PATTERN_TYPE['Literal']}>Literal</Radio>
<Radio value={ACL_PATTERN_TYPE['Prefixed']}>Prefixed</Radio> <Radio value={ACL_PATTERN_TYPE['Prefixed']}>Prefixed</Radio>
</Radio.Group> </Radio.Group>
</Form.Item> </Form.Item> */}
<Form.Item <Form.Item
label="Resource Type" label="Resource Type"
name="resourceType" name="resourceType"
@@ -428,7 +452,7 @@ const AddDrawer = forwardRef((_, ref) => {
<Form.Item dependencies={['resourceType']}> <Form.Item dependencies={['resourceType']}>
{({ getFieldValue }) => { {({ getFieldValue }) => {
const type = getFieldValue('resourceType'); const type = getFieldValue('resourceType');
if (type === ACL_RESOURCE_TYPE['Cluster'] || type === ACL_RESOURCE_TYPE['TransactionalId']) { if (type === ACL_RESOURCE_TYPE['Cluster']) {
//TODO需要和后端获取集群和事务接口联调 //TODO需要和后端获取集群和事务接口联调
return ( return (
<Form.Item <Form.Item
@@ -438,6 +462,8 @@ const AddDrawer = forwardRef((_, ref) => {
<Input placeholder={`请输入${type === 4 ? 'Cluster名称' : 'TransactionalId'}`}></Input> <Input placeholder={`请输入${type === 4 ? 'Cluster名称' : 'TransactionalId'}`}></Input>
</Form.Item> </Form.Item>
); );
} else if (type === ACL_RESOURCE_TYPE['TransactionalId']) {
return <PatternTypeFormItems type="transactionalId" />;
} else if (type === ACL_RESOURCE_TYPE['Topic']) { } else if (type === ACL_RESOURCE_TYPE['Topic']) {
return <PatternTypeFormItems type="topic" />; return <PatternTypeFormItems type="topic" />;
} else if (type === ACL_RESOURCE_TYPE['Group']) { } else if (type === ACL_RESOURCE_TYPE['Group']) {

View File

@@ -14,6 +14,7 @@ import AddACLDrawer, {
RESOURCE_TO_OPERATIONS_MAP, RESOURCE_TO_OPERATIONS_MAP,
RESOURCE_MAP_KEYS, RESOURCE_MAP_KEYS,
} from './EditDrawer'; } from './EditDrawer';
import { ClustersPermissionMap } from '../CommonConfig';
import './index.less'; import './index.less';
const { confirm } = Modal; const { confirm } = Modal;
@@ -105,7 +106,7 @@ const SecurityACLs = (): JSX.Element => {
}; };
const columns = () => { const columns = () => {
const baseColumns = [ const baseColumns: any = [
{ {
title: 'Principal', title: 'Principal',
dataIndex: 'kafkaUser', dataIndex: 'kafkaUser',
@@ -143,7 +144,9 @@ const SecurityACLs = (): JSX.Element => {
title: 'Host', title: 'Host',
dataIndex: 'aclClientHost', dataIndex: 'aclClientHost',
}, },
{ ];
if (global.hasPermission && global.hasPermission(ClustersPermissionMap.SECURITY_ACL_DELETE)) {
baseColumns.push({
title: '操作', title: '操作',
dataIndex: '', dataIndex: '',
width: 120, width: 120,
@@ -156,8 +159,8 @@ const SecurityACLs = (): JSX.Element => {
</> </>
); );
}, },
}, });
]; }
return baseColumns; return baseColumns;
}; };
@@ -238,15 +241,19 @@ const SecurityACLs = (): JSX.Element => {
</Form.Item> </Form.Item>
</Form> </Form>
</div> </div>
<div className={`${tableHeaderPrefix}-right`}> {global.hasPermission && global.hasPermission(ClustersPermissionMap.SECURITY_ACL_ADD) ? (
<Button <div className={`${tableHeaderPrefix}-right`}>
type="primary" <Button
// icon={<PlusOutlined />} type="primary"
onClick={() => editDrawerRef.current.onOpen(true, getACLs)} // icon={<PlusOutlined />}
> onClick={() => editDrawerRef.current.onOpen(true, getACLs)}
ACL >
</Button> ACL
</div> </Button>
</div>
) : (
<></>
)}
</div> </div>
<ProTable <ProTable
tableProps={{ tableProps={{

View File

@@ -82,7 +82,7 @@ export const getTopicMessagesColmns = () => {
dataIndex: 'offset', dataIndex: 'offset',
key: 'offset', key: 'offset',
sorter: true, sorter: true,
render: (t: number) => (+t ? t.toLocaleString() : '-'), render: (t: number) => (+t || +t === 0 ? t.toLocaleString() : '-'), // TODO: 千分位展示
}, },
{ {
title: 'Timestamp', title: 'Timestamp',

View File

@@ -299,7 +299,7 @@ public class ClusterMetricVersionItems extends BaseMetricVersionMetric {
// MessagesIn 指标 // MessagesIn 指标
itemList.add( buildAllVersionsItem() itemList.add( buildAllVersionsItem()
.name(CLUSTER_METRIC_MESSAGES_IN).unit("条/s").desc("集群每消息写入条数").category(CATEGORY_CLUSTER) .name(CLUSTER_METRIC_MESSAGES_IN).unit("条/s").desc("集群每消息写入条数").category(CATEGORY_CLUSTER)
.extend( buildJMXMethodExtend( CLUSTER_METHOD_GET_METRIC_FROM_KAFKA_BY_TOTAL_BROKERS_JMX ) .extend( buildJMXMethodExtend( CLUSTER_METHOD_GET_METRIC_FROM_KAFKA_BY_TOTAL_BROKERS_JMX )
.jmxObjectName( JMX_SERVER_BROKER_MESSAGES_IN ).jmxAttribute(RATE_MIN_1))); .jmxObjectName( JMX_SERVER_BROKER_MESSAGES_IN ).jmxAttribute(RATE_MIN_1)));