mirror of
https://github.com/whyour/qinglong.git
synced 2025-05-23 23:06:06 +08:00
parent
5a5f4b8065
commit
89ed8527d6
|
@ -28,6 +28,7 @@ export default (app: Router) => {
|
||||||
command: Joi.string().required(),
|
command: Joi.string().required(),
|
||||||
schedule: Joi.string().required(),
|
schedule: Joi.string().required(),
|
||||||
name: Joi.string().optional(),
|
name: Joi.string().optional(),
|
||||||
|
labels: Joi.array().optional(),
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
async (req: Request, res: Response, next: NextFunction) => {
|
async (req: Request, res: Response, next: NextFunction) => {
|
||||||
|
@ -83,6 +84,48 @@ export default (app: Router) => {
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
route.put(
|
||||||
|
'/removelabels',
|
||||||
|
celebrate({
|
||||||
|
body: Joi.object({
|
||||||
|
ids:Joi.array().items(Joi.number().required()),
|
||||||
|
labels:Joi.array().items(Joi.string().required()),
|
||||||
|
})
|
||||||
|
}),
|
||||||
|
async (req: Request, res: Response, next: NextFunction) => {
|
||||||
|
const logger: Logger = Container.get('logger');
|
||||||
|
try {
|
||||||
|
const cronService = Container.get(CronService);
|
||||||
|
const data = await cronService.removeLabels(req.body.ids,req.body.labels);
|
||||||
|
return res.send({ code: 200, data });
|
||||||
|
} catch (e) {
|
||||||
|
logger.error('🔥 error: %o', e);
|
||||||
|
return next(e);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
route.put(
|
||||||
|
'/addlabels',
|
||||||
|
celebrate({
|
||||||
|
body: Joi.object({
|
||||||
|
ids:Joi.array().items(Joi.number().required()),
|
||||||
|
labels:Joi.array().items(Joi.string().required()),
|
||||||
|
})
|
||||||
|
}),
|
||||||
|
async (req: Request, res: Response, next: NextFunction) => {
|
||||||
|
const logger: Logger = Container.get('logger');
|
||||||
|
try {
|
||||||
|
const cronService = Container.get(CronService);
|
||||||
|
const data = await cronService.addLabels(req.body.ids,req.body.labels);
|
||||||
|
return res.send({ code: 200, data });
|
||||||
|
} catch (e) {
|
||||||
|
logger.error('🔥 error: %o', e);
|
||||||
|
return next(e);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
route.put(
|
route.put(
|
||||||
'/disable',
|
'/disable',
|
||||||
celebrate({
|
celebrate({
|
||||||
|
@ -143,10 +186,11 @@ export default (app: Router) => {
|
||||||
'/',
|
'/',
|
||||||
celebrate({
|
celebrate({
|
||||||
body: Joi.object({
|
body: Joi.object({
|
||||||
|
labels: Joi.array().optional(),
|
||||||
command: Joi.string().optional(),
|
command: Joi.string().optional(),
|
||||||
schedule: Joi.string().optional(),
|
schedule: Joi.string().optional(),
|
||||||
name: Joi.string().optional(),
|
name: Joi.string().optional(),
|
||||||
id: Joi.string().required(),
|
id: Joi.number().required(),
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
async (req: Request, res: Response, next: NextFunction) => {
|
async (req: Request, res: Response, next: NextFunction) => {
|
||||||
|
|
|
@ -14,6 +14,7 @@ export class Crontab {
|
||||||
isDisabled?: 1 | 0;
|
isDisabled?: 1 | 0;
|
||||||
log_path?: string;
|
log_path?: string;
|
||||||
isPinned?: 1 | 0;
|
isPinned?: 1 | 0;
|
||||||
|
labels: Array<string>;
|
||||||
last_running_time?: number;
|
last_running_time?: number;
|
||||||
last_execution_time?: number;
|
last_execution_time?: number;
|
||||||
|
|
||||||
|
@ -33,6 +34,7 @@ export class Crontab {
|
||||||
this.isDisabled = options.isDisabled || 0;
|
this.isDisabled = options.isDisabled || 0;
|
||||||
this.log_path = options.log_path || '';
|
this.log_path = options.log_path || '';
|
||||||
this.isPinned = options.isPinned || 0;
|
this.isPinned = options.isPinned || 0;
|
||||||
|
this.labels = options.labels || [''];
|
||||||
this.last_running_time = options.last_running_time || 0;
|
this.last_running_time = options.last_running_time || 0;
|
||||||
this.last_execution_time = options.last_execution_time || 0;
|
this.last_execution_time = options.last_execution_time || 0;
|
||||||
}
|
}
|
||||||
|
@ -58,6 +60,18 @@ export const CrontabModel = sequelize.define<CronInstance>('Crontab', {
|
||||||
isDisabled: DataTypes.NUMBER,
|
isDisabled: DataTypes.NUMBER,
|
||||||
isPinned: DataTypes.NUMBER,
|
isPinned: DataTypes.NUMBER,
|
||||||
log_path: DataTypes.STRING,
|
log_path: DataTypes.STRING,
|
||||||
|
labels: {
|
||||||
|
type: DataTypes.STRING,
|
||||||
|
allowNull: false,
|
||||||
|
get() {
|
||||||
|
if (this.getDataValue('labels')) {
|
||||||
|
return this.getDataValue('labels').split(',')
|
||||||
|
}
|
||||||
|
},
|
||||||
|
set(value) {
|
||||||
|
this.setDataValue('labels', value.join(','));
|
||||||
|
},
|
||||||
|
},
|
||||||
last_running_time: DataTypes.NUMBER,
|
last_running_time: DataTypes.NUMBER,
|
||||||
last_execution_time: DataTypes.NUMBER,
|
last_execution_time: DataTypes.NUMBER,
|
||||||
});
|
});
|
||||||
|
|
|
@ -93,17 +93,43 @@ export default class CronService {
|
||||||
await CrontabModel.update({ isPinned: 0 }, { where: { id: ids } });
|
await CrontabModel.update({ isPinned: 0 }, { where: { id: ids } });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async addLabels(ids: string[],labels: string[]){
|
||||||
|
const docs = await CrontabModel.findAll({ where: { id:ids }});
|
||||||
|
for (const doc of docs) {
|
||||||
|
await CrontabModel.update({
|
||||||
|
labels: Array.from(new Set(doc.labels.concat(labels)))
|
||||||
|
},{ where: {id:doc.id}});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async removeLabels(ids: string[],labels: string[]){
|
||||||
|
const docs = await CrontabModel.findAll({ where: { id:ids }});
|
||||||
|
for (const doc of docs) {
|
||||||
|
await CrontabModel.update({
|
||||||
|
labels: doc.labels.filter( label => !labels.includes(label) )
|
||||||
|
},{ where: {id:doc.id}});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public async crontabs(searchText?: string): Promise<Crontab[]> {
|
public async crontabs(searchText?: string): Promise<Crontab[]> {
|
||||||
let query = {};
|
let query = {};
|
||||||
if (searchText) {
|
if (searchText) {
|
||||||
const encodeText = encodeURIComponent(searchText);
|
const textArray = searchText.split(":");
|
||||||
const reg = {
|
switch (textArray[0]) {
|
||||||
[Op.or]: [
|
case "name":
|
||||||
{ [Op.like]: `%${searchText}&` },
|
query = {name:{[Op.or]:createRegexp(textArray[1])}};
|
||||||
{ [Op.like]: `%${encodeText}%` },
|
break;
|
||||||
],
|
case "command":
|
||||||
};
|
query = {command:{[Op.or]:createRegexp(textArray[1])}};
|
||||||
|
break;
|
||||||
|
case "schedule":
|
||||||
|
query = {schedule:{[Op.or]:createRegexp(textArray[1])}};
|
||||||
|
break;
|
||||||
|
case "label":
|
||||||
|
query = {labels:{[Op.or]:createRegexp(textArray[1])}};
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
const reg = createRegexp(searchText);
|
||||||
query = {
|
query = {
|
||||||
[Op.or]: [
|
[Op.or]: [
|
||||||
{
|
{
|
||||||
|
@ -115,8 +141,13 @@ export default class CronService {
|
||||||
{
|
{
|
||||||
schedule: reg,
|
schedule: reg,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
labels: reg,
|
||||||
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
const result = await CrontabModel.findAll({ where: query });
|
const result = await CrontabModel.findAll({ where: query });
|
||||||
|
@ -124,6 +155,14 @@ export default class CronService {
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
|
function createRegexp(text:string) {
|
||||||
|
return {
|
||||||
|
[Op.or]: [
|
||||||
|
{ [Op.like]: `%${text}%` },
|
||||||
|
{ [Op.like]: `%${encodeURIComponent(text)}%` },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async get(id: number): Promise<Crontab> {
|
public async get(id: number): Promise<Crontab> {
|
||||||
|
|
|
@ -119,8 +119,8 @@ export GOBOT_QQ=""
|
||||||
## gotify_url 填写gotify地址,如https://push.example.de:8080
|
## gotify_url 填写gotify地址,如https://push.example.de:8080
|
||||||
## gotify_token 填写gotify的消息应用token
|
## gotify_token 填写gotify的消息应用token
|
||||||
## gotify_priority 填写推送消息优先级,默认为0
|
## gotify_priority 填写推送消息优先级,默认为0
|
||||||
export GOTIFY_URL="";
|
export GOTIFY_URL=""
|
||||||
export GOTIFY_TOKEN="";
|
export GOTIFY_TOKEN=""
|
||||||
export GOTIFY_PRIORITY=0;
|
export GOTIFY_PRIORITY=0
|
||||||
|
|
||||||
## 其他需要的变量,脚本中需要的变量使用 export 变量名= 声明即可
|
## 其他需要的变量,脚本中需要的变量使用 export 变量名= 声明即可
|
||||||
|
|
|
@ -11,6 +11,7 @@ import {
|
||||||
Menu,
|
Menu,
|
||||||
Typography,
|
Typography,
|
||||||
Input,
|
Input,
|
||||||
|
Popover,
|
||||||
} from 'antd';
|
} from 'antd';
|
||||||
import {
|
import {
|
||||||
ClockCircleOutlined,
|
ClockCircleOutlined,
|
||||||
|
@ -30,7 +31,7 @@ import {
|
||||||
import config from '@/utils/config';
|
import config from '@/utils/config';
|
||||||
import { PageContainer } from '@ant-design/pro-layout';
|
import { PageContainer } from '@ant-design/pro-layout';
|
||||||
import { request } from '@/utils/http';
|
import { request } from '@/utils/http';
|
||||||
import CronModal from './modal';
|
import CronModal,{ CronLabelModal } from './modal';
|
||||||
import CronLogModal from './logModal';
|
import CronLogModal from './logModal';
|
||||||
import cron_parser from 'cron-parser';
|
import cron_parser from 'cron-parser';
|
||||||
import { diffTime } from '@/utils/date';
|
import { diffTime } from '@/utils/date';
|
||||||
|
@ -77,12 +78,13 @@ const Crontab = ({ headerStyle, isPhone }: any) => {
|
||||||
width: 150,
|
width: 150,
|
||||||
align: 'center' as const,
|
align: 'center' as const,
|
||||||
render: (text: string, record: any) => (
|
render: (text: string, record: any) => (
|
||||||
|
<>
|
||||||
<a
|
<a
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
goToScriptManager(record);
|
goToScriptManager(record);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{record.name || record.id}{' '}
|
{record.name || record._id}{' '}
|
||||||
{record.isPinned ? (
|
{record.isPinned ? (
|
||||||
<span>
|
<span>
|
||||||
<PushpinOutlined />
|
<PushpinOutlined />
|
||||||
|
@ -91,6 +93,24 @@ const Crontab = ({ headerStyle, isPhone }: any) => {
|
||||||
''
|
''
|
||||||
)}
|
)}
|
||||||
</a>
|
</a>
|
||||||
|
<span>
|
||||||
|
{record.labels?.length > 0 && record.labels[0] !== '' ?
|
||||||
|
<Popover placement='right' trigger={isPhone ? 'click' : 'hover'}
|
||||||
|
content={
|
||||||
|
<div>
|
||||||
|
{record.labels?.map((label: string, i: number) => (
|
||||||
|
<Tag color="blue"
|
||||||
|
onClick={() => { onSearch(`label:${label}`) }}>
|
||||||
|
{label}
|
||||||
|
</Tag>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
}>
|
||||||
|
<Tag color="blue">{record.labels[0]}</Tag>
|
||||||
|
</Popover>
|
||||||
|
: ''}
|
||||||
|
</span>
|
||||||
|
</>
|
||||||
),
|
),
|
||||||
sorter: {
|
sorter: {
|
||||||
compare: (a: any, b: any) => a.name.localeCompare(b.name),
|
compare: (a: any, b: any) => a.name.localeCompare(b.name),
|
||||||
|
@ -325,6 +345,7 @@ const Crontab = ({ headerStyle, isPhone }: any) => {
|
||||||
const [value, setValue] = useState<any[]>([]);
|
const [value, setValue] = useState<any[]>([]);
|
||||||
const [loading, setLoading] = useState(true);
|
const [loading, setLoading] = useState(true);
|
||||||
const [isModalVisible, setIsModalVisible] = useState(false);
|
const [isModalVisible, setIsModalVisible] = useState(false);
|
||||||
|
const [isLabelModalVisible, setisLabelModalVisible] = useState(false);
|
||||||
const [editedCron, setEditedCron] = useState();
|
const [editedCron, setEditedCron] = useState();
|
||||||
const [searchText, setSearchText] = useState('');
|
const [searchText, setSearchText] = useState('');
|
||||||
const [isLogModalVisible, setIsLogModalVisible] = useState(false);
|
const [isLogModalVisible, setIsLogModalVisible] = useState(false);
|
||||||
|
@ -853,6 +874,13 @@ const Crontab = ({ headerStyle, isPhone }: any) => {
|
||||||
>
|
>
|
||||||
批量取消置顶
|
批量取消置顶
|
||||||
</Button>
|
</Button>
|
||||||
|
<Button
|
||||||
|
type="primary"
|
||||||
|
onClick={() => setisLabelModalVisible(true)}
|
||||||
|
style={{ marginLeft: 8, marginRight: 8 }}
|
||||||
|
>
|
||||||
|
批量修改标签
|
||||||
|
</Button>
|
||||||
<span style={{ marginLeft: 8 }}>
|
<span style={{ marginLeft: 8 }}>
|
||||||
已选择
|
已选择
|
||||||
<a>{selectedRowIds?.length}</a>项
|
<a>{selectedRowIds?.length}</a>项
|
||||||
|
@ -893,6 +921,16 @@ const Crontab = ({ headerStyle, isPhone }: any) => {
|
||||||
handleCancel={handleCancel}
|
handleCancel={handleCancel}
|
||||||
cron={editedCron}
|
cron={editedCron}
|
||||||
/>
|
/>
|
||||||
|
<CronLabelModal
|
||||||
|
visible={isLabelModalVisible}
|
||||||
|
handleCancel={(needUpdate?: boolean) => {
|
||||||
|
setisLabelModalVisible(false);
|
||||||
|
if (needUpdate) {
|
||||||
|
getCrons();
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
ids={selectedRowIds}
|
||||||
|
/>
|
||||||
</PageContainer>
|
</PageContainer>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import { Modal, message, Input, Form } from 'antd';
|
import { Modal, message, Input, Form, Button } from 'antd';
|
||||||
import { request } from '@/utils/http';
|
import { request } from '@/utils/http';
|
||||||
import config from '@/utils/config';
|
import config from '@/utils/config';
|
||||||
import cronParse from 'cron-parser';
|
import cronParse from 'cron-parser';
|
||||||
|
@ -48,6 +48,9 @@ const CronModal = ({
|
||||||
form
|
form
|
||||||
.validateFields()
|
.validateFields()
|
||||||
.then((values) => {
|
.then((values) => {
|
||||||
|
if (typeof values.labels === "string") {
|
||||||
|
values.labels = values.labels.split(/,|,/);
|
||||||
|
}
|
||||||
handleOk(values);
|
handleOk(values);
|
||||||
})
|
})
|
||||||
.catch((info) => {
|
.catch((info) => {
|
||||||
|
@ -66,6 +69,9 @@ const CronModal = ({
|
||||||
<Form.Item name="name" label="名称">
|
<Form.Item name="name" label="名称">
|
||||||
<Input placeholder="请输入任务名称" />
|
<Input placeholder="请输入任务名称" />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
|
<Form.Item name="labels" label="标签">
|
||||||
|
<Input placeholder="请输入任务标签" />
|
||||||
|
</Form.Item>
|
||||||
<Form.Item
|
<Form.Item
|
||||||
name="command"
|
name="command"
|
||||||
label="命令"
|
label="命令"
|
||||||
|
@ -100,4 +106,79 @@ const CronModal = ({
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default CronModal;
|
const CronLabelModal = ({
|
||||||
|
ids,
|
||||||
|
handleCancel,
|
||||||
|
visible,
|
||||||
|
}: {
|
||||||
|
ids: Array<string>;
|
||||||
|
visible: boolean;
|
||||||
|
handleCancel: (needUpdate?: boolean) => void;
|
||||||
|
}) => {
|
||||||
|
const [form] = Form.useForm();
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
|
||||||
|
const update = async (action: string) => {
|
||||||
|
form
|
||||||
|
.validateFields()
|
||||||
|
.then(async (values) => {
|
||||||
|
if (typeof values.labels === "string") {
|
||||||
|
values.labels = values.labels.split(/,|,/);
|
||||||
|
}
|
||||||
|
setLoading(true);
|
||||||
|
const payload = { ids, labels: values.labels };
|
||||||
|
const { code, data } = await request.put(`${config.apiPrefix}crons/${action}`, {
|
||||||
|
data: payload,
|
||||||
|
});
|
||||||
|
if (code === 200) {
|
||||||
|
message.success(action === 'addLabels' ? '添加Labels成功' : '删除Labels成功');
|
||||||
|
} else {
|
||||||
|
message.error(data);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
handleCancel(true);
|
||||||
|
})
|
||||||
|
.catch((info) => {
|
||||||
|
console.log('Validate Failed:', info);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
form.resetFields();
|
||||||
|
}, [ids, visible]);
|
||||||
|
|
||||||
|
const buttons = [
|
||||||
|
<Button onClick={() => handleCancel(false)} key="test">
|
||||||
|
取消
|
||||||
|
</Button>,
|
||||||
|
<Button type="primary" danger onClick={() => update('removeLabels')}>
|
||||||
|
删除
|
||||||
|
</Button>,
|
||||||
|
<Button type="primary" onClick={() => update('addLabels')}>
|
||||||
|
添加
|
||||||
|
</Button>
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
title='批量修改标签'
|
||||||
|
visible={visible}
|
||||||
|
footer={buttons}
|
||||||
|
forceRender
|
||||||
|
onCancel={() => handleCancel(false)}
|
||||||
|
confirmLoading={loading}
|
||||||
|
>
|
||||||
|
<Form
|
||||||
|
form={form}
|
||||||
|
layout="vertical"
|
||||||
|
name="form_in_label_modal"
|
||||||
|
>
|
||||||
|
<Form.Item name="labels" label="标签">
|
||||||
|
<Input placeholder="请输入任务标签" />
|
||||||
|
</Form.Item>
|
||||||
|
</Form>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export { CronModal as default, CronLabelModal }
|
||||||
|
|
|
@ -113,17 +113,19 @@ const Script = ({ headerStyle, isPhone, theme, socketMessage }: any) => {
|
||||||
const initGetScript = () => {
|
const initGetScript = () => {
|
||||||
const { p, s } = history.location.query as any;
|
const { p, s } = history.location.query as any;
|
||||||
if (s) {
|
if (s) {
|
||||||
|
const vkey = `${p}/${s}`;
|
||||||
const obj = {
|
const obj = {
|
||||||
node: {
|
node: {
|
||||||
title: s,
|
title: s,
|
||||||
value: s,
|
value: s,
|
||||||
key: p ? `${p}/${s}` : s,
|
key: p ? vkey : s,
|
||||||
parent: p,
|
parent: p,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
setExpandedKeys([p]);
|
setExpandedKeys([p]);
|
||||||
onTreeSelect([`${p}/${s}`], obj);
|
onTreeSelect([vkey], obj);
|
||||||
}
|
}
|
||||||
|
history.push('/script');
|
||||||
};
|
};
|
||||||
|
|
||||||
const onSelect = (value: any, node: any) => {
|
const onSelect = (value: any, node: any) => {
|
||||||
|
|
Loading…
Reference in New Issue
Block a user