Commit fe14f1dd authored by wenmo's avatar wenmo

chooseDB

parent ae0d397c
...@@ -5,6 +5,8 @@ import com.dlink.db.model.SuperEntity; ...@@ -5,6 +5,8 @@ import com.dlink.db.model.SuperEntity;
import lombok.Data; import lombok.Data;
import lombok.EqualsAndHashCode; import lombok.EqualsAndHashCode;
import java.time.LocalDateTime;
/** /**
* DataBase * DataBase
* *
...@@ -39,4 +41,8 @@ public class DataBase extends SuperEntity { ...@@ -39,4 +41,8 @@ public class DataBase extends SuperEntity {
private String dbVersion; private String dbVersion;
private boolean status; private boolean status;
private LocalDateTime healthTime;
private LocalDateTime heartbeatTime;
} }
...@@ -17,6 +17,8 @@ ...@@ -17,6 +17,8 @@
<result column="db_version" property="dbVersion" /> <result column="db_version" property="dbVersion" />
<result column="note" property="note" /> <result column="note" property="note" />
<result column="status" property="status" /> <result column="status" property="status" />
<result column="health_time" property="healthTime" />
<result column="heartbeat_time" property="heartbeatTime" />
<result column="enabled" property="enabled" /> <result column="enabled" property="enabled" />
<result column="create_time" property="createTime" /> <result column="create_time" property="createTime" />
<result column="update_time" property="updateTime" /> <result column="update_time" property="updateTime" />
...@@ -24,7 +26,7 @@ ...@@ -24,7 +26,7 @@
<!-- 通用查询结果列 --> <!-- 通用查询结果列 -->
<sql id="Base_Column_List"> <sql id="Base_Column_List">
id, name, alias,group_name, type,ip,port, url,username,password,db_version,note,status, enabled, create_time, update_time id, name, alias,group_name, type,ip,port, url,username,password,db_version,note,status,health_time,heartbeat_time, enabled, create_time, update_time
</sql> </sql>
......
...@@ -349,6 +349,8 @@ CREATE TABLE `dlink_database` ( ...@@ -349,6 +349,8 @@ CREATE TABLE `dlink_database` (
`note` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '注释', `note` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '注释',
`db_version` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '版本,如oracle的11g,hbase的2.2.3', `db_version` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '版本,如oracle的11g,hbase的2.2.3',
`status` tinyint(1) NULL COMMENT '状态', `status` tinyint(1) NULL COMMENT '状态',
`health_time` datetime(0) NULL DEFAULT NULL COMMENT '最近健康时间',
`heartbeat_time` datetime(0) NULL DEFAULT NULL COMMENT '最近心跳检测时间',
`enabled` tinyint(1) NOT NULL DEFAULT 1 COMMENT '启用', `enabled` tinyint(1) NOT NULL DEFAULT 1 COMMENT '启用',
`create_time` datetime(0) NULL DEFAULT NULL COMMENT '创建时间', `create_time` datetime(0) NULL DEFAULT NULL COMMENT '创建时间',
`update_time` datetime(0) NULL DEFAULT NULL COMMENT '最近修改时间', `update_time` datetime(0) NULL DEFAULT NULL COMMENT '最近修改时间',
......
...@@ -26,10 +26,6 @@ const SessionForm: React.FC<UpdateFormProps> = (props) => { ...@@ -26,10 +26,6 @@ const SessionForm: React.FC<UpdateFormProps> = (props) => {
type: props.values.type, type: props.values.type,
useRemote: props.values.useRemote, useRemote: props.values.useRemote,
clusterId: props.values.clusterId, clusterId: props.values.clusterId,
clusterName: props.values.clusterName,
address: props.values.address,
createUser: props.values.createUser,
createTime: props.values.createTime,
}); });
const {cluster} = props; const {cluster} = props;
......
export function getDBImage(type:string) {
let imageUrl = '/database/';
switch (type.toLowerCase()){
case 'mysql':
imageUrl += 'mysql.jpg';
break;
case 'oracle':
imageUrl += 'oracle.jpg';
break;
case 'postgresql':
imageUrl += 'postgresql.jpg';
break;
case 'clickhouse':
imageUrl += 'clickhouse.png';
break;
default:
imageUrl += 'db.png';
}
return imageUrl;
}
import React, {useState} from 'react';
import {Modal,Image,message,List,Card} from 'antd';
import {DataBaseItem} from '../data.d';
import {connect} from "umi";
import {StateType} from "@/pages/FlinkSqlStudio/model";
import {getDBImage} from "@/pages/DataBase/DB";
import MysqlForm from "@/pages/DataBase/components/MySqlForm";
export type UpdateFormProps = {
onCancel: (flag?: boolean, formVals?: Partial<DataBaseItem>) => void;
onSubmit: (values: Partial<DataBaseItem>) => void;
modalVisible: boolean;
values: Partial<DataBaseItem>;
};
const data = [
{
type: 'MySql',
},
{
type: 'Oracle',
},
{
type: 'PostgreSql',
},
{
type: 'ClickHouse',
},
];
const ChooseDB: React.FC<UpdateFormProps> = (props) => {
const {
// onSubmit: handleUpdate,
onCancel: handleChooseDBModalVisible,
modalVisible,
} = props;
const [dbType, setDbType] = useState<string>();
const [createModalVisible, handleModalVisible] = useState<boolean>(false);
const chooseOne = (item:DataBaseItem)=>{
setDbType(item.type);
handleModalVisible(true);
// handleChooseDBModalVisible();
};
return (
<Modal
width={800}
bodyStyle={{padding: '32px 40px 48px'}}
title={'选择数据源类型'}
visible={modalVisible}
onCancel={() => handleChooseDBModalVisible()}
footer={null}
>
<List
grid={{
gutter: 16,
xs: 1,
sm: 2,
md: 4,
lg: 4,
xl: 4,
xxl: 4,
}}
dataSource={data}
renderItem={item => (
<List.Item onClick={()=>{chooseOne(item)}}>
<Card>
<Image
height={60}
preview={false}
src={getDBImage(item.type)}
fallback=""
/>
</Card>
</List.Item>
)}
/>
<MysqlForm
onCancel={() => setDbType(undefined)}
modalVisible={dbType=='MySql'}
values={{}}
onSubmit={()=>{
setDbType(undefined)
}}
/>
</Modal>
);
};
export default connect(({Studio}: { Studio: StateType }) => ({
cluster: Studio.cluster,
}))(ChooseDB);
import React, {useEffect, useState} from 'react';
import { Form, Button, Input, Modal,Select } from 'antd';
import Switch from "antd/es/switch";
import TextArea from "antd/es/input/TextArea";
import {DataBaseItem} from "@/pages/DataBase/data";
export type MysqlFormProps = {
onCancel: (flag?: boolean, formVals?: Partial<DataBaseItem>) => void;
onSubmit: (values: Partial<DataBaseItem>) => void;
modalVisible: boolean;
values: Partial<DataBaseItem>;
};
const FormItem = Form.Item;
const formLayout = {
labelCol: { span: 7 },
wrapperCol: { span: 13 },
};
const MysqlForm: React.FC<MysqlFormProps> = (props) => {
const [formVals, setFormVals] = useState<Partial<DataBaseItem>>({
id: props.values.id,
name: props.values.name,
alias: props.values.alias,
type: props.values.type,
note: props.values.note,
enabled: props.values.enabled,
});
const [form] = Form.useForm();
const {
onSubmit: handleUpdate,
onCancel: handleUpdateModalVisible,
modalVisible,
values,
} = props;
const submitForm = async () => {
const fieldsValue = await form.validateFields();
setFormVals({ ...formVals, ...fieldsValue });
handleUpdate({ ...formVals, ...fieldsValue });
};
const renderContent = (formVals) => {
return (
<>
<FormItem
name="name"
label="名称"
rules={[{ required: true, message: '请输入名称!' }]} >
<Input placeholder="请输入" />
</FormItem>
<FormItem
name="alias"
label="别名"
>
<Input placeholder="请输入" />
</FormItem>
<FormItem
name="type"
label="类型"
>
<Select defaultValue="Yarn" allowClear>
<Option value="Standalone">Standalone</Option>
<Option value="Yarn">Yarn</Option>
<Option value="Others">Others</Option>
</Select>
</FormItem>
<FormItem
name="hosts"
label="Hosts"
>
<TextArea placeholder="添加 Flink Hosts...例如:127.0.0.1:8081,127.0.0.1:8091" allowClear autoSize={{ minRows: 3, maxRows: 10 }}/>
</FormItem>
<FormItem
name="note"
label="注释"
>
<Input placeholder="请输入" />
</FormItem>
<FormItem
name="enabled"
label="是否启用"
rules={[{ required: true, message: '请输入是否启用!' }]} >
<Switch checkedChildren="启用" unCheckedChildren="禁用"
defaultChecked={formVals.enabled}/>
</FormItem>
</>
);
};
const renderFooter = () => {
return (
<>
<Button onClick={() => handleUpdateModalVisible(false, values)}>取消</Button>
<Button type="primary" onClick={() => submitForm()}>
完成
</Button>
</>
);
};
return (
<Modal
width={640}
bodyStyle={{ padding: '32px 40px 48px' }}
destroyOnClose
title={formVals.id?'编辑':'创建 MySql 数据源'}
visible={modalVisible}
footer={renderFooter()}
onCancel={() => handleUpdateModalVisible()}
>
<Form
{...formLayout}
form={form}
initialValues={{
id: formVals.id,
name: formVals.name,
alias: formVals.alias,
type: formVals.type,
note: formVals.note,
enabled: formVals.enabled,
}}
>
{renderContent(formVals)}
</Form>
</Modal>
);
};
export default MysqlForm;
export type DataBaseTableListItem = { export type DataBaseItem = {
id: number, id: number,
name: string, name: string,
alias: string, alias: string,
...@@ -12,7 +12,25 @@ export type DataBaseTableListItem = { ...@@ -12,7 +12,25 @@ export type DataBaseTableListItem = {
note: string, note: string,
dbVersion: string, dbVersion: string,
status: boolean, status: boolean,
healthTime: Date,
heartbeatTime: Date,
enabled: boolean, enabled: boolean,
createTime: Date, createTime: Date,
updateTime: Date, updateTime: Date,
}; };
export type DataBaseFormProps = {
name: string,
alias: string,
groupName: string,
type: string,
ip: string,
port: number,
url: string,
username: string,
password: string,
note: string,
dbVersion: string,
enabled: boolean,
}
import React from "react"; import React, {useState} from "react";
import {PageContainer, FooterToolbar} from '@ant-design/pro-layout'; import {PageContainer, FooterToolbar} from '@ant-design/pro-layout';
import {DownOutlined, HeartOutlined, PlusOutlined, UserOutlined} from '@ant-design/icons'; import {DownOutlined, HeartOutlined, PlusOutlined, UserOutlined} from '@ant-design/icons';
import { Progress, Tag, Button,Space,Badge,Typography } from 'antd'; import {Progress, Tag, Button, Space, Badge, Typography, Image, Row, Col} from 'antd';
import ProList from '@ant-design/pro-list'; import ProList from '@ant-design/pro-list';
import {queryData} from "@/components/Common/crud"; import {queryData} from "@/components/Common/crud";
import {getDBImage} from "@/pages/DataBase/DB";
import ChooseDB from "./components/ChooseDB";
const { Text } = Typography; const {Text} = Typography;
const url = '/api/database'; const url = '/api/database';
const DataBaseTableList: React.FC<{}> = (props: any) => { const DataBaseTableList: React.FC<{}> = (props: any) => {
const [chooseDBModalVisible, handleChooseDBModalVisible] = useState<boolean>(false);
const chooseDB = () =>{
};
return ( return (
<PageContainer> <PageContainer>
<ProList <ProList
toolBarRender={() => { toolBarRender={() => {
return [ return [
<Button key="3" type="primary"> <Button type="primary" onClick={() => handleChooseDBModalVisible(true)}>
<PlusOutlined/> 新建 <PlusOutlined/> 新建
</Button>, </Button>,
]; ];
}} }}
pagination={{ pagination={{
defaultPageSize: 8, defaultPageSize: 8,
showSizeChanger: false, showSizeChanger: false,
}} }}
grid={{ gutter: 16, column: 4 }} grid={{gutter: 16, column: 4}}
request={(params, sorter, filter) => queryData(url,{...params, sorter:{id:'descend'}, filter})} request={(params, sorter, filter) => queryData(url, {...params, sorter: {id: 'descend'}, filter})}
metas={{ metas={{
title: { title: {
dataIndex: 'alias', dataIndex: 'alias',
title: 'alias', title: 'alias',
}, },
subTitle: { subTitle: {
render: (_, row) => { render: (_, row) => {
return ( return (
<Space size={0}> <Space size={0}>
<Tag color="blue" key={row.name}> <Tag color="blue" key={row.name}>
{row.name} {row.name}
</Tag> </Tag>
<Tag color="gray" key={row.groupName}> <Tag color="gray" key={row.groupName}>
{row.groupName} {row.groupName}
</Tag> </Tag>
{(row.status) ? {(row.status) ?
(<><Badge status="success"/><Text type="success">正常</Text></>): (<><Badge status="success"/><Text type="success">正常</Text></>) :
<><Badge status="error"/><Text type="danger">异常</Text></>} <><Badge status="error"/><Text type="danger">异常</Text></>}
</Space> </Space>
); );
},
}, },
}, type: {},
type: {}, avatar: {},
avatar: {}, content: {
content: {}, render: (_, row) => {
actions: {}, return (
}} <Row justify="space-around" align="middle">
<Col span={16}>
<Image
height={120}
preview={false}
src={getDBImage(row.type)}
fallback=""
/>
</Col>
</Row>
)
},
},
actions: {},
}}
headerTitle="数据源" headerTitle="数据源"
// dataSource={data} />
/> <ChooseDB onCancel={() => handleChooseDBModalVisible(false)} modalVisible={chooseDBModalVisible}/>
</PageContainer> </PageContainer>
); );
}; };
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment