mirror of
https://github.com/1Panel-dev/1Panel.git
synced 2025-01-19 16:29:17 +08:00
feat: 计划任务备份应用/网站/数据库支持多选 (#5849)
This commit is contained in:
parent
cbb119117b
commit
f6c334bdad
@ -17,15 +17,20 @@ import (
|
|||||||
|
|
||||||
func (u *CronjobService) handleApp(cronjob model.Cronjob, startTime time.Time) error {
|
func (u *CronjobService) handleApp(cronjob model.Cronjob, startTime time.Time) error {
|
||||||
var apps []model.AppInstall
|
var apps []model.AppInstall
|
||||||
if cronjob.AppID == "all" {
|
if strings.Contains(cronjob.AppID, "all") {
|
||||||
apps, _ = appInstallRepo.ListBy()
|
apps, _ = appInstallRepo.ListBy()
|
||||||
} else {
|
} else {
|
||||||
itemID, _ := strconv.Atoi(cronjob.AppID)
|
appIds := strings.Split(cronjob.AppID, ",")
|
||||||
app, err := appInstallRepo.GetFirst(commonRepo.WithByID(uint(itemID)))
|
var idItems []uint
|
||||||
|
for i := 0; i < len(appIds); i++ {
|
||||||
|
itemID, _ := strconv.Atoi(appIds[i])
|
||||||
|
idItems = append(idItems, uint(itemID))
|
||||||
|
}
|
||||||
|
appItems, err := appInstallRepo.ListBy(commonRepo.WithIdsIn(idItems))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
apps = append(apps, app)
|
apps = appItems
|
||||||
}
|
}
|
||||||
accountMap, err := loadClientMap(cronjob.BackupAccounts)
|
accountMap, err := loadClientMap(cronjob.BackupAccounts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -232,7 +237,7 @@ type databaseHelper struct {
|
|||||||
|
|
||||||
func loadDbsForJob(cronjob model.Cronjob) []databaseHelper {
|
func loadDbsForJob(cronjob model.Cronjob) []databaseHelper {
|
||||||
var dbs []databaseHelper
|
var dbs []databaseHelper
|
||||||
if cronjob.DBName == "all" {
|
if strings.Contains(cronjob.DBName, "all") {
|
||||||
if cronjob.DBType == "mysql" || cronjob.DBType == "mariadb" {
|
if cronjob.DBType == "mysql" || cronjob.DBType == "mariadb" {
|
||||||
mysqlItems, _ := mysqlRepo.List()
|
mysqlItems, _ := mysqlRepo.List()
|
||||||
for _, mysql := range mysqlItems {
|
for _, mysql := range mysqlItems {
|
||||||
@ -254,7 +259,10 @@ func loadDbsForJob(cronjob model.Cronjob) []databaseHelper {
|
|||||||
}
|
}
|
||||||
return dbs
|
return dbs
|
||||||
}
|
}
|
||||||
itemID, _ := strconv.Atoi(cronjob.DBName)
|
|
||||||
|
dbNames := strings.Split(cronjob.DBName, ",")
|
||||||
|
for _, name := range dbNames {
|
||||||
|
itemID, _ := strconv.Atoi(name)
|
||||||
if cronjob.DBType == "mysql" || cronjob.DBType == "mariadb" {
|
if cronjob.DBType == "mysql" || cronjob.DBType == "mariadb" {
|
||||||
mysqlItem, _ := mysqlRepo.Get(commonRepo.WithByID(uint(itemID)))
|
mysqlItem, _ := mysqlRepo.Get(commonRepo.WithByID(uint(itemID)))
|
||||||
dbs = append(dbs, databaseHelper{
|
dbs = append(dbs, databaseHelper{
|
||||||
@ -270,20 +278,23 @@ func loadDbsForJob(cronjob model.Cronjob) []databaseHelper {
|
|||||||
Name: pgItem.Name,
|
Name: pgItem.Name,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return dbs
|
return dbs
|
||||||
}
|
}
|
||||||
|
|
||||||
func loadWebsForJob(cronjob model.Cronjob) []model.Website {
|
func loadWebsForJob(cronjob model.Cronjob) []model.Website {
|
||||||
var weblist []model.Website
|
var weblist []model.Website
|
||||||
if cronjob.Website == "all" {
|
if strings.Contains(cronjob.Website, "all") {
|
||||||
weblist, _ = websiteRepo.List()
|
weblist, _ = websiteRepo.List()
|
||||||
return weblist
|
return weblist
|
||||||
}
|
}
|
||||||
itemID, _ := strconv.Atoi(cronjob.Website)
|
websites := strings.Split(cronjob.Website, ",")
|
||||||
webItem, _ := websiteRepo.GetFirst(commonRepo.WithByID(uint(itemID)))
|
var idItems []uint
|
||||||
if webItem.ID != 0 {
|
for i := 0; i < len(websites); i++ {
|
||||||
weblist = append(weblist, webItem)
|
itemID, _ := strconv.Atoi(websites[i])
|
||||||
|
idItems = append(idItems, uint(itemID))
|
||||||
}
|
}
|
||||||
|
weblist, _ = websiteRepo.List(commonRepo.WithIdsIn(idItems))
|
||||||
return weblist
|
return weblist
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,6 +24,9 @@ export namespace Cronjob {
|
|||||||
backupAccounts: string;
|
backupAccounts: string;
|
||||||
defaultDownload: string;
|
defaultDownload: string;
|
||||||
backupAccountList: Array<string>;
|
backupAccountList: Array<string>;
|
||||||
|
appIdList: Array<string>;
|
||||||
|
websiteList: Array<string>;
|
||||||
|
dbNameList: Array<string>;
|
||||||
retainCopies: number;
|
retainCopies: number;
|
||||||
status: string;
|
status: string;
|
||||||
secret: string;
|
secret: string;
|
||||||
|
@ -186,9 +186,9 @@
|
|||||||
<el-form-item
|
<el-form-item
|
||||||
v-if="dialogData.rowData!.type === 'website' || dialogData.rowData!.type === 'cutWebsiteLog'"
|
v-if="dialogData.rowData!.type === 'website' || dialogData.rowData!.type === 'cutWebsiteLog'"
|
||||||
:label="dialogData.rowData!.type === 'website' ? $t('cronjob.website'):$t('website.website')"
|
:label="dialogData.rowData!.type === 'website' ? $t('cronjob.website'):$t('website.website')"
|
||||||
prop="website"
|
prop="websiteList"
|
||||||
>
|
>
|
||||||
<el-select class="selectClass" v-model="dialogData.rowData!.website">
|
<el-select class="selectClass" multiple v-model="dialogData.rowData!.websiteList">
|
||||||
<el-option
|
<el-option
|
||||||
:disabled="websiteOptions.length === 0"
|
:disabled="websiteOptions.length === 0"
|
||||||
:label="$t('commons.table.all')"
|
:label="$t('commons.table.all')"
|
||||||
@ -212,8 +212,8 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<div v-if="dialogData.rowData!.type === 'app'">
|
<div v-if="dialogData.rowData!.type === 'app'">
|
||||||
<el-form-item :label="$t('cronjob.app')" prop="appID">
|
<el-form-item :label="$t('cronjob.app')" prop="appIdList">
|
||||||
<el-select class="selectClass" clearable v-model="dialogData.rowData!.appID">
|
<el-select class="selectClass" multiple clearable v-model="dialogData.rowData!.appIdList">
|
||||||
<el-option
|
<el-option
|
||||||
:disabled="appOptions.length === 0"
|
:disabled="appOptions.length === 0"
|
||||||
:label="$t('commons.table.all')"
|
:label="$t('commons.table.all')"
|
||||||
@ -239,8 +239,8 @@
|
|||||||
<el-radio value="postgresql">PostgreSQL</el-radio>
|
<el-radio value="postgresql">PostgreSQL</el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item :label="$t('cronjob.database')" prop="dbName">
|
<el-form-item :label="$t('cronjob.database')" prop="dbNameList">
|
||||||
<el-select class="selectClass" clearable v-model="dialogData.rowData!.dbName">
|
<el-select class="selectClass" multiple clearable v-model="dialogData.rowData!.dbNameList">
|
||||||
<el-option
|
<el-option
|
||||||
:disabled="dbInfo.dbs.length === 0"
|
:disabled="dbInfo.dbs.length === 0"
|
||||||
:label="$t('commons.table.all')"
|
:label="$t('commons.table.all')"
|
||||||
@ -378,6 +378,7 @@ import { listContainer } from '@/api/modules/container';
|
|||||||
import { Database } from '@/api/interface/database';
|
import { Database } from '@/api/interface/database';
|
||||||
import { ListAppInstalled } from '@/api/modules/app';
|
import { ListAppInstalled } from '@/api/modules/app';
|
||||||
import { loadDefaultSpec, specOptions, transObjToSpec, transSpecToObj, weekOptions } from './../helper';
|
import { loadDefaultSpec, specOptions, transObjToSpec, transSpecToObj, weekOptions } from './../helper';
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
interface DialogProps {
|
interface DialogProps {
|
||||||
@ -385,6 +386,7 @@ interface DialogProps {
|
|||||||
rowData?: Cronjob.CronjobInfo;
|
rowData?: Cronjob.CronjobInfo;
|
||||||
getTableList?: () => Promise<any>;
|
getTableList?: () => Promise<any>;
|
||||||
}
|
}
|
||||||
|
|
||||||
const title = ref<string>('');
|
const title = ref<string>('');
|
||||||
const drawerVisible = ref(false);
|
const drawerVisible = ref(false);
|
||||||
const dialogData = ref<DialogProps>({
|
const dialogData = ref<DialogProps>({
|
||||||
@ -408,6 +410,15 @@ const acceptParams = (params: DialogProps): void => {
|
|||||||
if (dialogData.value.rowData.backupAccounts) {
|
if (dialogData.value.rowData.backupAccounts) {
|
||||||
dialogData.value.rowData.backupAccountList = dialogData.value.rowData.backupAccounts.split(',');
|
dialogData.value.rowData.backupAccountList = dialogData.value.rowData.backupAccounts.split(',');
|
||||||
}
|
}
|
||||||
|
if (dialogData.value.rowData.appID) {
|
||||||
|
dialogData.value.rowData.appIdList = dialogData.value.rowData.appID.split(',');
|
||||||
|
}
|
||||||
|
if (dialogData.value.rowData.website) {
|
||||||
|
dialogData.value.rowData.websiteList = dialogData.value.rowData.website.split(',');
|
||||||
|
}
|
||||||
|
if (dialogData.value.rowData.dbName) {
|
||||||
|
dialogData.value.rowData.dbNameList = dialogData.value.rowData.dbName.split(',');
|
||||||
|
}
|
||||||
dialogData.value.rowData!.command = dialogData.value.rowData!.command || 'sh';
|
dialogData.value.rowData!.command = dialogData.value.rowData!.command || 'sh';
|
||||||
dialogData.value.rowData!.isCustom =
|
dialogData.value.rowData!.isCustom =
|
||||||
dialogData.value.rowData!.command !== 'sh' &&
|
dialogData.value.rowData!.command !== 'sh' &&
|
||||||
@ -718,6 +729,17 @@ const onSubmit = async (formEl: FormInstance | undefined) => {
|
|||||||
specs.push(itemSpec);
|
specs.push(itemSpec);
|
||||||
}
|
}
|
||||||
dialogData.value.rowData.backupAccounts = dialogData.value.rowData.backupAccountList.join(',');
|
dialogData.value.rowData.backupAccounts = dialogData.value.rowData.backupAccountList.join(',');
|
||||||
|
|
||||||
|
if (dialogData.value.rowData.appIdList) {
|
||||||
|
dialogData.value.rowData.appID = dialogData.value.rowData.appIdList.join(',');
|
||||||
|
}
|
||||||
|
if (dialogData.value.rowData.websiteList) {
|
||||||
|
dialogData.value.rowData.website = dialogData.value.rowData.websiteList.join(',');
|
||||||
|
}
|
||||||
|
if (dialogData.value.rowData.dbNameList) {
|
||||||
|
dialogData.value.rowData.dbName = dialogData.value.rowData.dbNameList.join(',');
|
||||||
|
}
|
||||||
|
|
||||||
dialogData.value.rowData.spec = specs.join(',');
|
dialogData.value.rowData.spec = specs.join(',');
|
||||||
if (!formEl) return;
|
if (!formEl) return;
|
||||||
formEl.validate(async (valid) => {
|
formEl.validate(async (valid) => {
|
||||||
@ -750,40 +772,49 @@ defineExpose({
|
|||||||
.specClass {
|
.specClass {
|
||||||
width: 20% !important;
|
width: 20% !important;
|
||||||
margin-left: 20px;
|
margin-left: 20px;
|
||||||
|
|
||||||
.append {
|
.append {
|
||||||
width: 20px;
|
width: 20px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media only screen and (max-width: 1000px) {
|
@media only screen and (max-width: 1000px) {
|
||||||
.specClass {
|
.specClass {
|
||||||
width: 100% !important;
|
width: 100% !important;
|
||||||
margin-top: 20px;
|
margin-top: 20px;
|
||||||
margin-left: 0;
|
margin-left: 0;
|
||||||
|
|
||||||
.append {
|
.append {
|
||||||
width: 43px;
|
width: 43px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.specTypeClass {
|
.specTypeClass {
|
||||||
width: 22% !important;
|
width: 22% !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media only screen and (max-width: 1000px) {
|
@media only screen and (max-width: 1000px) {
|
||||||
.specTypeClass {
|
.specTypeClass {
|
||||||
width: 100% !important;
|
width: 100% !important;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.selectClass {
|
.selectClass {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tagClass {
|
.tagClass {
|
||||||
float: right;
|
float: right;
|
||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
margin-top: 5px;
|
margin-top: 5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.logText {
|
.logText {
|
||||||
line-height: 22px;
|
line-height: 22px;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
|
|
||||||
.link {
|
.link {
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
margin-top: -3px;
|
margin-top: -3px;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user