1
0
mirror of https://github.com/1Panel-dev/1Panel.git synced 2025-01-19 16:29:17 +08:00

fix: 计划任务记录样式调整

This commit is contained in:
ssongliu 2023-02-07 10:29:55 +08:00 committed by ssongliu
parent ae9c6e8e61
commit 83b9607eb7
6 changed files with 344 additions and 281 deletions

View File

@ -2,10 +2,8 @@ package service
import ( import (
"context" "context"
"crypto/tls"
"fmt" "fmt"
"io/ioutil" "io/ioutil"
"net/http"
"os" "os"
"os/exec" "os/exec"
"strings" "strings"
@ -15,6 +13,7 @@ import (
"github.com/1Panel-dev/1Panel/backend/constant" "github.com/1Panel-dev/1Panel/backend/constant"
"github.com/1Panel-dev/1Panel/backend/global" "github.com/1Panel-dev/1Panel/backend/global"
"github.com/1Panel-dev/1Panel/backend/utils/cloud_storage" "github.com/1Panel-dev/1Panel/backend/utils/cloud_storage"
"github.com/1Panel-dev/1Panel/backend/utils/cmd"
"github.com/pkg/errors" "github.com/pkg/errors"
"github.com/robfig/cron/v3" "github.com/robfig/cron/v3"
) )
@ -28,12 +27,14 @@ func (u *CronjobService) HandleJob(cronjob *model.Cronjob) {
record.FromLocal = cronjob.KeepLocal record.FromLocal = cronjob.KeepLocal
switch cronjob.Type { switch cronjob.Type {
case "shell": case "shell":
cmd := exec.Command(cronjob.Script) if len(cronjob.Script) == 0 {
stdout, errExec := cmd.CombinedOutput() return
if errExec != nil {
err = errors.New(string(stdout))
} }
message = stdout stdout, errExec := cmd.Exec(cronjob.Script)
if errExec != nil {
err = errExec
}
message = []byte(stdout)
case "website": case "website":
record.File, err = u.HandleBackup(cronjob, record.StartTime) record.File, err = u.HandleBackup(cronjob, record.StartTime)
case "database": case "database":
@ -47,17 +48,11 @@ func (u *CronjobService) HandleJob(cronjob *model.Cronjob) {
if len(cronjob.URL) == 0 { if len(cronjob.URL) == 0 {
return return
} }
tr := &http.Transport{ stdout, errCurl := cmd.Exec("curl " + cronjob.URL)
TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
}
client := &http.Client{Timeout: 1 * time.Second, Transport: tr}
request, _ := http.NewRequest("GET", cronjob.URL, nil)
response, err := client.Do(request)
if err != nil { if err != nil {
cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), string(message)) err = errCurl
} }
defer response.Body.Close() message = []byte(stdout)
message, _ = ioutil.ReadAll(response.Body)
} }
if err != nil { if err != nil {
cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), string(message)) cronjobRepo.EndRecords(record, constant.StatusFailed, err.Error(), string(message))

View File

@ -19,6 +19,9 @@ func Init() {
panic(err) panic(err)
} }
baseDir := strings.ReplaceAll(stdout, "\n", "") baseDir := strings.ReplaceAll(stdout, "\n", "")
if len(baseDir) == 0 {
panic("error `BASE_DIR` find in /usr/bin/1pctl")
}
v := viper.NewWithOptions() v := viper.NewWithOptions()
v.SetConfigType("yaml") v.SetConfigType("yaml")
reader := bytes.NewReader(conf.AppYaml) reader := bytes.NewReader(conf.AppYaml)

View File

@ -499,12 +499,14 @@ export default {
disableMsg: 'The cronjob cannot continue to run after it is stopped. Do you want to stop it?', disableMsg: 'The cronjob cannot continue to run after it is stopped. Do you want to stop it?',
enableMsg: 'The cronjob has been stopped. Enable now?', enableMsg: 'The cronjob has been stopped. Enable now?',
taskType: 'Task type', taskType: 'Task type',
record: 'Records',
shell: 'shell', shell: 'shell',
website: 'website', website: 'website',
rulesHelper: 'Compression exclusion rules (with; Is a delimiter), for example: \n*.log; *.sql', rulesHelper: 'Compression exclusion rules (with; Is a delimiter), for example: \n*.log; *.sql',
lastRecrodTime: 'Last execution time', lastRecrodTime: 'Last execution time',
failedFilter: 'Failed Task Filtering', all: 'All',
all: 'all', failedRecord: 'Failed records',
successRecord: 'Successful records',
database: 'database', database: 'database',
missBackupAccount: 'The backup account could not be found', missBackupAccount: 'The backup account could not be found',
syncDate: 'Synchronization time ', syncDate: 'Synchronization time ',

View File

@ -523,12 +523,14 @@ export default {
disableMsg: '计划任务停止后将无法继续运行是否停止', disableMsg: '计划任务停止后将无法继续运行是否停止',
enableMsg: '该计划任务已停止是否启用', enableMsg: '该计划任务已停止是否启用',
taskType: '任务类型', taskType: '任务类型',
record: '报告',
shell: 'Shell 脚本', shell: 'Shell 脚本',
website: '备份网站', website: '备份网站',
rulesHelper: '压缩排除规则( ; 号为分隔符)例如 \n*.log;*.sql', rulesHelper: '压缩排除规则( ; 号为分隔符)例如 \n*.log;*.sql',
failedFilter: '失败任务过滤',
lastRecrodTime: '上次执行时间', lastRecrodTime: '上次执行时间',
all: '所有', all: '所有',
failedRecord: '失败任务',
successRecord: '成功任务',
database: '备份数据库', database: '备份数据库',
missBackupAccount: '未能找到备份账号', missBackupAccount: '未能找到备份账号',
syncDate: '同步时间 ', syncDate: '同步时间 ',

View File

@ -8,7 +8,7 @@
}, },
]" ]"
/> />
<LayoutContent v-loading="loading" :title="$t('cronjob.cronTask')"> <LayoutContent v-loading="loading" v-if="!isRecordShow" :title="$t('cronjob.cronTask')">
<template #toolbar> <template #toolbar>
<el-button type="primary" @click="onOpenDialog('create')"> <el-button type="primary" @click="onOpenDialog('create')">
{{ $t('commons.button.create') }}{{ $t('cronjob.cronTask') }} {{ $t('commons.button.create') }}{{ $t('cronjob.cronTask') }}
@ -69,13 +69,18 @@
{{ $t('cronjob.handle') }} {{ $t('cronjob.handle') }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column :label="$t('cronjob.retainCopies')" prop="retainCopies" /> <el-table-column :label="$t('cronjob.retainCopies')" :width="90" prop="retainCopies">
<template #default="{ row }">
{{ loadCopies(row) }}
</template>
</el-table-column>
<el-table-column :label="$t('cronjob.lastRecrodTime')" prop="lastRecrodTime"> <el-table-column :label="$t('cronjob.lastRecrodTime')" prop="lastRecrodTime">
<template #default="{ row }"> <template #default="{ row }">
{{ row.lastRecrodTime }} {{ row.lastRecrodTime }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column :label="$t('cronjob.target')" prop="targetDir"> <el-table-column :width="80" :label="$t('cronjob.target')" prop="targetDir">
<template #default="{ row }"> <template #default="{ row }">
{{ loadBackupName(row.targetDir) }} {{ loadBackupName(row.targetDir) }}
</template> </template>
@ -92,14 +97,14 @@
</LayoutContent> </LayoutContent>
<OperatrDialog @search="search" ref="dialogRef" /> <OperatrDialog @search="search" ref="dialogRef" />
<RecordDialog ref="dialogRecordRef" /> <Records ref="dialogRecordRef" />
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import ComplexTable from '@/components/complex-table/index.vue'; import ComplexTable from '@/components/complex-table/index.vue';
import OperatrDialog from '@/views/cronjob/operate/index.vue'; import OperatrDialog from '@/views/cronjob/operate/index.vue';
import RecordDialog from '@/views/cronjob/record/index.vue'; import Records from '@/views/cronjob/record/index.vue';
import LayoutContent from '@/layout/layout-content.vue'; import LayoutContent from '@/layout/layout-content.vue';
import { loadZero } from '@/utils/util'; import { loadZero } from '@/utils/util';
import { onMounted, reactive, ref } from 'vue'; import { onMounted, reactive, ref } from 'vue';
@ -113,6 +118,7 @@ import { ElMessage, ElMessageBox } from 'element-plus';
const loading = ref(); const loading = ref();
const selects = ref<any>([]); const selects = ref<any>([]);
const isRecordShow = ref();
const data = ref(); const data = ref();
const paginationConfig = reactive({ const paginationConfig = reactive({
@ -153,12 +159,9 @@ const search = async () => {
}); });
}; };
const dialogRecordRef = ref<DialogExpose>(); const dialogRecordRef = ref();
interface DialogExpose { const dialogRef = ref();
acceptParams: (params: any) => void;
}
const dialogRef = ref<DialogExpose>();
const onOpenDialog = async ( const onOpenDialog = async (
title: string, title: string,
rowData: Partial<Cronjob.CronjobInfo> = { rowData: Partial<Cronjob.CronjobInfo> = {
@ -204,9 +207,24 @@ const onChangeStatus = async (id: number, status: string) => {
}; };
const onHandle = async (row: Cronjob.CronjobInfo) => { const onHandle = async (row: Cronjob.CronjobInfo) => {
await handleOnce(row.id); loading.value = true;
ElMessage.success(i18n.global.t('commons.msg.operationSuccess')); await handleOnce(row.id)
search(); .then(() => {
loading.value = false;
ElMessage.success(i18n.global.t('commons.msg.operationSuccess'));
search();
})
.catch(() => {
loading.value = false;
});
};
const loadCopies = (item) => {
if (item.type === 'shell' || item.type === 'curl') {
return '-';
} else {
return item.retainCopies + '';
}
}; };
const buttons = [ const buttons = [
@ -232,20 +250,17 @@ const buttons = [
}, },
}, },
{ {
label: i18n.global.t('commons.button.view'), label: i18n.global.t('cronjob.record'),
icon: 'Clock', icon: 'Clock',
click: (row: Cronjob.CronjobInfo) => { click: (row: Cronjob.CronjobInfo) => {
onOpenRecordDialog(row); isRecordShow.value = true;
let params = {
rowData: { ...row },
};
dialogRecordRef.value!.acceptParams(params);
}, },
}, },
]; ];
const onOpenRecordDialog = async (rowData: Partial<Cronjob.CronjobInfo> = {}) => {
let params = {
rowData: { ...rowData },
};
dialogRecordRef.value!.acceptParams(params);
};
function loadWeek(i: number) { function loadWeek(i: number) {
for (const week of weekOptions) { for (const week of weekOptions) {
if (week.value === i) { if (week.value === i) {

View File

@ -1,229 +1,233 @@
<template> <template>
<el-dialog v-model="cronjobVisiable" :destroy-on-close="true" :close-on-click-modal="false" width="70%"> <div v-if="recordShow" v-loading="loading">
<template #header> <div class="a-card" style="margin-top: 20px">
<div class="card-header"> <el-card>
<span>{{ title }}{{ $t('cronjob.cronTask') }}</span> <div>
</div> <el-tag style="float: left" effect="dark" type="success">{{ dialogData.rowData.name }}</el-tag>
</template> <el-tag round class="status-content" type="success">
<el-date-picker {{ $t('cronjob.' + dialogData.rowData.type) }}
@change="search()" </el-tag>
v-model="timeRangeLoad" <el-tag class="status-content" type="info">
type="datetimerange" <span
:range-separator="$t('commons.search.timeRange')" v-if="
:start-placeholder="$t('commons.search.timeStart')" dialogData.rowData?.specType.indexOf('N') === -1 ||
:end-placeholder="$t('commons.search.timeEnd')" dialogData.rowData?.specType === 'perWeek'
:shortcuts="shortcuts" "
></el-date-picker>
<el-checkbox style="margin-left: 20px" @change="search()" v-model="searchInfo.status">
{{ $t('cronjob.failedFilter') }}
</el-checkbox>
<el-row :gutter="20" style="margin-top: 20px">
<el-col :span="6">
<el-card>
<ul v-infinite-scroll="nextPage" class="infinite-list" style="overflow: auto">
<li
v-for="(item, index) in records"
:key="index"
@click="forDetail(item)"
class="infinite-list-item"
> >
<el-icon v-if="item.status === 'Success'"><Select /></el-icon> {{ $t('cronjob.' + dialogData.rowData?.specType) }}&nbsp;
<el-icon v-if="item.status === 'Failed'"><CloseBold /></el-icon> </span>
{{ dateFormatSimple(item.startTime) }} <span v-else>{{ $t('cronjob.per') }}</span>
</li> <span v-if="dialogData.rowData?.specType === 'perMonth'">
</ul> {{ dialogData.rowData?.day }}{{ $t('cronjob.day') }}&nbsp;
<div style="margin-top: 10px; margin-bottom: 5px; font-size: 12px; float: right"> {{ loadZero(dialogData.rowData?.hour) }} :
<span>{{ $t('commons.table.total', [searchInfo.recordTotal]) }}</span> {{ loadZero(dialogData.rowData?.minute) }}
</div> </span>
</el-card> <span v-if="dialogData.rowData?.specType === 'perWeek'">
</el-col> {{ loadWeek(dialogData.rowData?.week) }}&nbsp; {{ loadZero(dialogData.rowData?.hour) }} :
<el-col :span="18"> {{ loadZero(dialogData.rowData?.minute) }}
<el-card style="height: 352px"> </span>
<el-form> <span v-if="dialogData.rowData?.specType === 'perNDay'">
<el-row> {{ dialogData.rowData?.day }}{{ $t('cronjob.day1') }},&nbsp;
<el-col :span="8"> {{ loadZero(dialogData.rowData?.hour) }} :
<el-form-item :label="$t('cronjob.taskType')"> {{ loadZero(dialogData.rowData?.minute) }}
{{ dialogData.rowData?.type }} </span>
</el-form-item> <span v-if="dialogData.rowData?.specType === 'perNHour'">
</el-col> {{ dialogData.rowData?.hour }}{{ $t('cronjob.hour') }},&nbsp;
<el-col :span="8"> {{ loadZero(dialogData.rowData?.minute) }}
<el-form-item :label="$t('cronjob.taskName')"> </span>
{{ dialogData.rowData?.name }} <span v-if="dialogData.rowData?.specType === 'perHour'">
</el-form-item> &nbsp;{{ loadZero(dialogData.rowData?.minute) }}
</el-col> </span>
<el-col :span="8"> <span v-if="dialogData.rowData?.specType === 'perNMinute'">
<el-form-item :label="$t('cronjob.cronSpec')"> &nbsp;{{ dialogData.rowData?.minute }}{{ $t('cronjob.minute') }}
<span </span>
v-if=" &nbsp;{{ $t('cronjob.handle') }}
dialogData.rowData?.specType.indexOf('N') === -1 || </el-tag>
dialogData.rowData?.specType === 'perWeek' <span class="buttons">
" <el-button type="primary" @click="onHandle(dialogData.rowData)" link>
> {{ $t('commons.button.handle') }}
{{ $t('cronjob.' + dialogData.rowData?.specType) }}&nbsp; </el-button>
</span>
</div>
</el-card>
</div>
<LayoutContent :title="$t('cronjob.record')" :reload="true">
<template #main>
<el-row :gutter="20">
<el-col :span="6">
<el-date-picker
style="width: calc(100% - 20px)"
@change="search()"
v-model="timeRangeLoad"
type="datetimerange"
:range-separator="$t('commons.search.timeRange')"
:start-placeholder="$t('commons.search.timeStart')"
:end-placeholder="$t('commons.search.timeEnd')"
:shortcuts="shortcuts"
></el-date-picker>
<el-card style="margin-top: 20px">
<ul v-infinite-scroll="nextPage" class="infinite-list" style="overflow: auto">
<li
v-for="(item, index) in records"
:key="index"
@click="forDetail(item, index)"
class="infinite-list-item"
>
<el-icon v-if="item.status === 'Success'"><Select /></el-icon>
<el-icon v-if="item.status === 'Failed'"><CloseBold /></el-icon>
<span v-if="index === currentRecordIndex" style="color: red">
{{ dateFormat(0, 0, item.startTime) }}
</span> </span>
<span v-else>{{ $t('cronjob.per') }}</span> <span v-else>{{ dateFormat(0, 0, item.startTime) }}</span>
<span v-if="dialogData.rowData?.specType === 'perMonth'"> </li>
{{ dialogData.rowData?.day }}{{ $t('cronjob.day') }}&nbsp; </ul>
{{ loadZero(dialogData.rowData?.hour) }} : <div style="margin-top: 10px; margin-bottom: 5px; font-size: 12px; float: right">
{{ loadZero(dialogData.rowData?.minute) }} <span>{{ $t('commons.table.total', [searchInfo.recordTotal]) }}</span>
</span> </div>
<span v-if="dialogData.rowData?.specType === 'perWeek'"> </el-card>
{{ loadWeek(dialogData.rowData?.week) }}&nbsp; </el-col>
{{ loadZero(dialogData.rowData?.hour) }} : <el-col :span="18">
{{ loadZero(dialogData.rowData?.minute) }} <el-select @change="search()" v-model="searchInfo.status">
</span> <el-option :label="$t('cronjob.all')" value="" />
<span v-if="dialogData.rowData?.specType === 'perNDay'"> <el-option :label="$t('cronjob.failedRecord')" value="Failed" />
{{ dialogData.rowData?.day }}{{ $t('cronjob.day1') }},&nbsp; <el-option :label="$t('cronjob.successRecord')" value="Success" />
{{ loadZero(dialogData.rowData?.hour) }} : </el-select>
{{ loadZero(dialogData.rowData?.minute) }} <el-card style="height: 382px; margin-top: 20px">
</span> <el-form>
<span v-if="dialogData.rowData?.specType === 'perNHour'"> <el-row v-if="hasScript()">
{{ dialogData.rowData?.hour }}{{ $t('cronjob.hour') }},&nbsp; <span>{{ $t('cronjob.shellContent') }}</span>
{{ loadZero(dialogData.rowData?.minute) }} <codemirror
</span> ref="mymirror"
<span v-if="dialogData.rowData?.specType === 'perHour'"> :autofocus="true"
&nbsp;{{ loadZero(dialogData.rowData?.minute) }} placeholder="None data"
</span> :indent-with-tab="true"
<span v-if="dialogData.rowData?.specType === 'perNMinute'"> :tabSize="4"
&nbsp;{{ dialogData.rowData?.minute }}{{ $t('cronjob.minute') }} style="height: 120px; width: 100%; margin-top: 5px"
</span> :lineWrapping="true"
&nbsp;{{ $t('cronjob.handle') }} :matchBrackets="true"
</el-form-item> theme="cobalt"
</el-col> :styleActiveLine="true"
<el-col :span="8" v-if="hasScript()"> :extensions="extensions"
<el-form-item :label="$t('cronjob.shellContent')"> v-model="dialogData.rowData!.script"
<el-popover :readOnly="true"
placement="right" />
:width="600" </el-row>
trigger="click" <el-row>
style="white-space: pre-wrap" <el-col :span="8" v-if="dialogData.rowData!.type === 'website'">
> <el-form-item :label="$t('cronjob.website')">
<div style="margin-left: 20px; max-height: 400px; overflow: auto"> {{ dialogData.rowData!.website }}
<span style="white-space: pre-wrap">{{ dialogData.rowData!.script }}</span> </el-form-item>
</div> </el-col>
<template #reference> <el-col :span="8" v-if="dialogData.rowData!.type === 'database'">
<el-button type="primary" link>{{ $t('commons.button.expand') }}</el-button> <el-form-item :label="$t('cronjob.database')">
</template> {{ dialogData.rowData!.dbName }}
</el-popover> </el-form-item>
</el-form-item> </el-col>
</el-col> <el-col :span="8" v-if="dialogData.rowData!.type === 'directory'">
<el-col :span="8" v-if="dialogData.rowData!.type === 'website'"> <el-form-item :label="$t('cronjob.directory')">
<el-form-item :label="$t('cronjob.website')"> {{ dialogData.rowData!.sourceDir }}
{{ dialogData.rowData!.website }} </el-form-item>
</el-form-item> </el-col>
</el-col> <el-col :span="8" v-if="isBackup()">
<el-col :span="8" v-if="dialogData.rowData!.type === 'database'"> <el-form-item :label="$t('cronjob.target')">
<el-form-item :label="$t('cronjob.database')"> {{ loadBackupName(dialogData.rowData!.targetDir) }}
{{ dialogData.rowData!.dbName }} <el-button
</el-form-item> v-if="currentRecord?.status! !== 'Failed'"
</el-col> type="primary"
<el-col :span="8" v-if="dialogData.rowData!.type === 'directory'"> style="margin-left: 10px"
<el-form-item :label="$t('cronjob.directory')"> link
{{ dialogData.rowData!.sourceDir }} icon="Download"
</el-form-item> @click="onDownload(currentRecord!.id, dialogData.rowData!.targetDirID)"
</el-col> >
<el-col :span="8" v-if="isBackup()"> {{ $t('file.download') }}
<el-form-item :label="$t('cronjob.target')">
{{ loadBackupName(dialogData.rowData!.targetDir) }}
<el-button
v-if="currentRecord?.status! !== 'Failed'"
type="primary"
style="margin-left: 10px"
link
icon="Download"
@click="onDownload(currentRecord!.id, dialogData.rowData!.targetDirID)"
>
{{ $t('file.download') }}
</el-button>
</el-form-item>
</el-col>
<el-col :span="8" v-if="isBackup()">
<el-form-item :label="$t('cronjob.retainCopies')">
{{ dialogData.rowData!.retainCopies }}
</el-form-item>
</el-col>
<el-col :span="8" v-if="dialogData.rowData!.type === 'curl'">
<el-form-item :label="$t('cronjob.url')">
{{ dialogData.rowData!.url }}
</el-form-item>
</el-col>
<el-col
:span="8"
v-if="dialogData.rowData!.type === 'website' || dialogData.rowData!.type === 'directory'"
>
<el-form-item :label="$t('cronjob.exclusionRules')">
<div v-if="dialogData.rowData!.exclusionRules">
<div v-for="item in dialogData.rowData!.exclusionRules.split(';')" :key="item">
<el-tag>{{ item }}</el-tag>
</div>
</div>
<span v-else>-</span>
</el-form-item>
</el-col>
</el-row>
<el-row>
<el-col :span="8">
<el-form-item :label="$t('commons.search.timeStart')">
{{ dateFormatSimple(currentRecord?.startTime) }}
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item :label="$t('commons.table.interval')">
<span v-if="currentRecord?.interval! <= 1000">
{{ currentRecord?.interval }} ms
</span>
<span v-if="currentRecord?.interval! > 1000">
{{ currentRecord?.interval! / 1000 }} s
</span>
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item :label="$t('commons.table.status')">
<el-tooltip
v-if="currentRecord?.status === 'Failed'"
class="box-item"
:content="currentRecord?.message"
placement="top"
>
{{ $t('commons.table.statusFailed') }}
</el-tooltip>
<span v-else>{{ $t('commons.table.statusSuccess') }}</span>
</el-form-item>
</el-col>
</el-row>
<el-row v-if="currentRecord?.records">
<el-col :span="24">
<el-form-item :label="$t('commons.table.records')">
<el-popover
placement="right"
:width="600"
trigger="click"
style="white-space: pre-wrap"
>
<div style="margin-left: 20px; max-height: 400px; overflow: auto">
<span style="white-space: pre-wrap">
{{ currentRecordDetail }}
</span>
</div>
<template #reference>
<el-button type="primary" link @click="loadRecord(currentRecord?.records!)">
{{ $t('commons.button.expand') }}
</el-button> </el-button>
</template> </el-form-item>
</el-popover> </el-col>
</el-form-item> <el-col :span="8" v-if="isBackup()">
</el-col> <el-form-item :label="$t('cronjob.retainCopies')">
</el-row> {{ dialogData.rowData!.retainCopies }}
</el-form> </el-form-item>
</el-card> </el-col>
</el-col> <el-col :span="8" v-if="dialogData.rowData!.type === 'curl'">
</el-row> <el-form-item :label="$t('cronjob.url')">
<template #footer> {{ dialogData.rowData!.url }}
<span class="dialog-footer"> </el-form-item>
<el-button @click="cronjobVisiable = false">{{ $t('commons.button.cancel') }}</el-button> </el-col>
</span> <el-col
</template> :span="8"
</el-dialog> v-if="dialogData.rowData!.type === 'website' || dialogData.rowData!.type === 'directory'"
>
<el-form-item :label="$t('cronjob.exclusionRules')">
<div v-if="dialogData.rowData!.exclusionRules">
<div
v-for="item in dialogData.rowData!.exclusionRules.split(';')"
:key="item"
>
<el-tag>{{ item }}</el-tag>
</div>
</div>
<span v-else>-</span>
</el-form-item>
</el-col>
</el-row>
<el-row>
<el-col :span="8">
<el-form-item :label="$t('commons.search.timeStart')">
{{ dateFormat(0, 0, currentRecord?.startTime) }}
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item :label="$t('commons.table.interval')">
<span v-if="currentRecord?.interval! <= 1000">
{{ currentRecord?.interval }} ms
</span>
<span v-if="currentRecord?.interval! > 1000">
{{ currentRecord?.interval! / 1000 }} s
</span>
</el-form-item>
</el-col>
<el-col :span="8">
<el-form-item :label="$t('commons.table.status')">
<el-tooltip
v-if="currentRecord?.status === 'Failed'"
class="box-item"
:content="currentRecord?.message"
placement="top"
>
<el-tag type="danger">{{ $t('commons.table.statusFailed') }}</el-tag>
</el-tooltip>
<el-tag type="success" v-else>
{{ $t('commons.table.statusSuccess') }}
</el-tag>
</el-form-item>
</el-col>
</el-row>
<el-row v-if="currentRecord?.records">
<span>{{ $t('commons.table.records') }}</span>
<codemirror
ref="mymirror"
:autofocus="true"
placeholder="None data"
:indent-with-tab="true"
:tabSize="4"
style="height: 130px; width: 100%; margin-top: 5px"
:lineWrapping="true"
:matchBrackets="true"
theme="cobalt"
:styleActiveLine="true"
:extensions="extensions"
v-model="currentRecordDetail"
:readOnly="true"
/>
</el-row>
</el-form>
</el-card>
</el-col>
</el-row>
</template>
</LayoutContent>
</div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
@ -231,21 +235,30 @@ import { reactive, ref } from 'vue';
import { Cronjob } from '@/api/interface/cronjob'; import { Cronjob } from '@/api/interface/cronjob';
import { loadZero } from '@/utils/util'; import { loadZero } from '@/utils/util';
import { loadBackupName } from '@/views/setting/helper'; import { loadBackupName } from '@/views/setting/helper';
import { searchRecords, download } from '@/api/modules/cronjob'; import { searchRecords, download, handleOnce } from '@/api/modules/cronjob';
import { dateFormatSimple, dateFormatForName } from '@/utils/util'; import { dateFormat, dateFormatForName } from '@/utils/util';
import i18n from '@/lang'; import i18n from '@/lang';
import { ElMessage } from 'element-plus'; import { ElMessage } from 'element-plus';
import { LoadFile } from '@/api/modules/files'; import { LoadFile } from '@/api/modules/files';
import LayoutContent from '@/layout/layout-content.vue';
import { Codemirror } from 'vue-codemirror';
import { javascript } from '@codemirror/lang-javascript';
import { oneDark } from '@codemirror/theme-one-dark';
const loading = ref();
const mymirror = ref();
const extensions = [javascript(), oneDark];
interface DialogProps { interface DialogProps {
rowData?: Cronjob.CronjobInfo; rowData: Cronjob.CronjobInfo;
} }
const title = ref<string>(''); const recordShow = ref(false);
const cronjobVisiable = ref(false); const dialogData = ref();
const dialogData = ref<DialogProps>({});
const records = ref<Array<Cronjob.Record>>(); const records = ref<Array<Cronjob.Record>>();
const currentRecord = ref<Cronjob.Record>(); const currentRecord = ref<Cronjob.Record>();
const currentRecordDetail = ref<string>(''); const currentRecordDetail = ref<string>('');
const currentRecordIndex = ref();
const acceptParams = async (params: DialogProps): Promise<void> => { const acceptParams = async (params: DialogProps): Promise<void> => {
dialogData.value = params; dialogData.value = params;
@ -255,7 +268,7 @@ const acceptParams = async (params: DialogProps): Promise<void> => {
cronjobID: dialogData.value.rowData!.id, cronjobID: dialogData.value.rowData!.id,
startTime: new Date(new Date().setHours(0, 0, 0, 0)), startTime: new Date(new Date().setHours(0, 0, 0, 0)),
endTime: new Date(), endTime: new Date(),
status: searchInfo.status ? 'Stoped' : '', status: searchInfo.status,
}; };
const res = await searchRecords(itemSearch); const res = await searchRecords(itemSearch);
records.value = res.data.items || []; records.value = res.data.items || [];
@ -264,8 +277,10 @@ const acceptParams = async (params: DialogProps): Promise<void> => {
return; return;
} }
currentRecord.value = records.value[0]; currentRecord.value = records.value[0];
currentRecordIndex.value = 0;
loadRecord(currentRecord.value.records);
searchInfo.recordTotal = res.data.total; searchInfo.recordTotal = res.data.total;
cronjobVisiable.value = true; recordShow.value = true;
}; };
const shortcuts = [ const shortcuts = [
@ -320,17 +335,30 @@ const weekOptions = [
{ label: i18n.global.t('cronjob.saturday'), value: 6 }, { label: i18n.global.t('cronjob.saturday'), value: 6 },
{ label: i18n.global.t('cronjob.sunday'), value: 7 }, { label: i18n.global.t('cronjob.sunday'), value: 7 },
]; ];
const timeRangeLoad = ref<Array<any>>([new Date(new Date().setHours(0, 0, 0, 0)), new Date()]); const timeRangeLoad = ref<[Date, Date]>([new Date(new Date().setHours(0, 0, 0, 0)), new Date()]);
const searchInfo = reactive({ const searchInfo = reactive({
page: 1, page: 1,
pageSize: 10, pageSize: 8,
recordTotal: 0, recordTotal: 0,
cronjobID: 0, cronjobID: 0,
startTime: new Date(new Date().setHours(0, 0, 0, 0)), startTime: new Date(new Date().setHours(0, 0, 0, 0)),
endTime: new Date(), endTime: new Date(),
status: false, status: '',
}); });
const onHandle = async (row: Cronjob.CronjobInfo) => {
loading.value = true;
await handleOnce(row.id)
.then(() => {
loading.value = false;
ElMessage.success(i18n.global.t('commons.msg.operationSuccess'));
search();
})
.catch(() => {
loading.value = false;
});
};
const search = async () => { const search = async () => {
if (timeRangeLoad.value && timeRangeLoad.value.length === 2) { if (timeRangeLoad.value && timeRangeLoad.value.length === 2) {
searchInfo.startTime = timeRangeLoad.value[0]; searchInfo.startTime = timeRangeLoad.value[0];
@ -345,7 +373,7 @@ const search = async () => {
cronjobID: dialogData.value.rowData!.id, cronjobID: dialogData.value.rowData!.id,
startTime: searchInfo.startTime, startTime: searchInfo.startTime,
endTime: searchInfo.endTime, endTime: searchInfo.endTime,
status: searchInfo.status ? 'Failed' : '', status: searchInfo.status,
}; };
const res = await searchRecords(params); const res = await searchRecords(params);
records.value = res.data.items || []; records.value = res.data.items || [];
@ -374,15 +402,19 @@ const nextPage = async () => {
if (searchInfo.pageSize >= searchInfo.recordTotal) { if (searchInfo.pageSize >= searchInfo.recordTotal) {
return; return;
} }
searchInfo.pageSize = searchInfo.pageSize + 3; searchInfo.pageSize = searchInfo.pageSize + 5;
search(); search();
}; };
const forDetail = async (row: Cronjob.Record) => { const forDetail = async (row: Cronjob.Record, index: number) => {
currentRecord.value = row; currentRecord.value = row;
currentRecordIndex.value = index;
loadRecord(row.records);
}; };
const loadRecord = async (path: string) => { const loadRecord = async (path: string) => {
const res = await LoadFile({ path: path }); if (path) {
currentRecordDetail.value = res.data; const res = await LoadFile({ path: path });
currentRecordDetail.value = res.data;
}
}; };
function isBackup() { function isBackup() {
return ( return (
@ -408,9 +440,9 @@ defineExpose({
}); });
</script> </script>
<style scoped> <style lang="scss" scoped>
.infinite-list { .infinite-list {
height: 300px; height: 330px;
padding: 0; padding: 0;
margin: 0; margin: 0;
list-style: none; list-style: none;
@ -435,4 +467,18 @@ defineExpose({
font-weight: 500; font-weight: 500;
color: red; color: red;
} }
.a-card {
font-size: 17px;
.el-card {
--el-card-padding: 12px;
.buttons {
margin-left: 100px;
}
}
}
.status-content {
float: left;
margin-left: 50px;
}
</style> </style>