From 8fb989eac04a01b7b34d7b30665fb3aeb7020143 Mon Sep 17 00:00:00 2001 From: ssongliu Date: Thu, 23 Nov 2023 18:59:13 +0800 Subject: [PATCH 1/6] =?UTF-8?q?feat:=20=E5=A2=9E=E5=8A=A0=E6=9C=8D?= =?UTF-8?q?=E5=8A=A1=E5=99=A8=20Swap=20=E5=88=86=E5=8C=BA=E7=AE=A1?= =?UTF-8?q?=E7=90=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- backend/app/api/v1/device.go | 22 ++ backend/app/dto/device.go | 18 ++ backend/app/service/device.go | 117 +++++++++ backend/router/ro_toolbox.go | 1 + frontend/src/api/interface/toolbox.ts | 16 ++ frontend/src/api/modules/toolbox.ts | 4 + frontend/src/lang/modules/zh.ts | 3 + .../views/container/network/create/index.vue | 42 +++- frontend/src/views/home/status/index.vue | 2 +- frontend/src/views/toolbox/device/index.vue | 37 +++ .../src/views/toolbox/device/swap/index.vue | 234 ++++++++++++++++++ frontend/vite.config.ts | 1 - 12 files changed, 493 insertions(+), 4 deletions(-) create mode 100644 frontend/src/views/toolbox/device/swap/index.vue diff --git a/backend/app/api/v1/device.go b/backend/app/api/v1/device.go index 2769d73ca086..fc9ee3ca49f4 100644 --- a/backend/app/api/v1/device.go +++ b/backend/app/api/v1/device.go @@ -159,6 +159,28 @@ func (b *BaseApi) UpdateDevicPasswd(c *gin.Context) { helper.SuccessWithData(c, nil) } +// @Tags Device +// @Summary Update device swap +// @Description 修改系统 Swap +// @Accept json +// @Param request body dto.SwapHelper true "request" +// @Success 200 +// @Security ApiKeyAuth +// @Router /toolbox/device/update/swap [post] +// @x-panel-log {"bodyKeys":["opreate","path"],"paramKeys":[],"BeforeFunctions":[],"formatZH":"[opreate] 主机 swap [path]","formatEN":"[opreate] device swap [path]"} +func (b *BaseApi) UpdateDevicSwap(c *gin.Context) { + var req dto.SwapHelper + if err := helper.CheckBindAndValidate(&req, c); err != nil { + return + } + if err := deviceService.UpdateSwap(req); err != nil { + helper.ErrorWithDetail(c, constant.CodeErrInternalServer, constant.ErrTypeInternalServer, err) + return + } + + helper.SuccessWithData(c, nil) +} + // @Tags Device // @Summary Check device DNS conf // @Description 检查系统 DNS 配置可用性 diff --git a/backend/app/dto/device.go b/backend/app/dto/device.go index 3cbc4f84362a..b5b3616181c6 100644 --- a/backend/app/dto/device.go +++ b/backend/app/dto/device.go @@ -8,6 +8,15 @@ type DeviceBaseInfo struct { LocalTime string `json:"localTime"` Ntp string `json:"ntp"` User string `json:"user"` + + MemoryTotal uint64 `json:"memoryTotal"` + MemoryAvailable uint64 `json:"memoryAvailable"` + MemoryUsed uint64 `json:"memoryUsed"` + SwapMemoryTotal uint64 `json:"swapMemoryTotal"` + SwapMemoryAvailable uint64 `json:"swapMemoryAvailable"` + SwapMemoryUsed uint64 `json:"swapMemoryUsed"` + + SwapDetails []SwapHelper `json:"swapDetails"` } type HostHelper struct { @@ -15,6 +24,15 @@ type HostHelper struct { Host string `json:"host"` } +type SwapHelper struct { + Operate string `json:"operate" validate:"required,oneof=create delete update"` + Path string `json:"path" validate:"required"` + Size uint64 `json:"size"` + Used string `json:"used"` + + WithRemove bool `json:"withRemove"` +} + type TimeZoneOptions struct { From string `json:"from"` Zones []string `json:"zones"` diff --git a/backend/app/service/device.go b/backend/app/service/device.go index 5f779e2be46f..668620d4c70d 100644 --- a/backend/app/service/device.go +++ b/backend/app/service/device.go @@ -6,6 +6,7 @@ import ( "fmt" "net" "os" + "strconv" "strings" "time" @@ -14,10 +15,12 @@ import ( "github.com/1Panel-dev/1Panel/backend/utils/cmd" "github.com/1Panel-dev/1Panel/backend/utils/common" "github.com/1Panel-dev/1Panel/backend/utils/ntp" + "github.com/shirou/gopsutil/v3/mem" ) const defaultDNSPath = "/etc/resolv.conf" const defaultHostPath = "/etc/hosts" +const defaultFstab = "/etc/fstab" type DeviceService struct{} @@ -26,6 +29,7 @@ type IDeviceService interface { Update(key, value string) error UpdateHosts(req []dto.HostHelper) error UpdatePasswd(req dto.ChangePasswd) error + UpdateSwap(req dto.SwapHelper) error UpdateByConf(req dto.UpdateByNameAndFile) error LoadTimeZone() ([]string, error) CheckDNS(key, value string) (bool, error) @@ -47,6 +51,18 @@ func (u *DeviceService) LoadBaseInfo() (dto.DeviceBaseInfo, error) { ntp, _ := settingRepo.Get(settingRepo.WithByKey("NtpSite")) baseInfo.Ntp = ntp.Value + memoryInfo, _ := mem.VirtualMemory() + baseInfo.MemoryTotal = memoryInfo.Total + baseInfo.MemoryAvailable = memoryInfo.Available + baseInfo.MemoryUsed = memoryInfo.Used + swapInfo, _ := mem.SwapMemory() + baseInfo.SwapMemoryTotal = swapInfo.Total + baseInfo.SwapMemoryAvailable = swapInfo.Free + baseInfo.SwapMemoryUsed = swapInfo.Used + if baseInfo.SwapMemoryTotal != 0 { + baseInfo.SwapDetails = loadSwap() + } + return baseInfo, nil } @@ -168,6 +184,50 @@ func (u *DeviceService) UpdatePasswd(req dto.ChangePasswd) error { return nil } +func (u *DeviceService) UpdateSwap(req dto.SwapHelper) error { + if req.Operate != "create" { + std, err := cmd.Execf("%s swapoff %s", cmd.SudoHandleCmd(), req.Path) + if err != nil { + return fmt.Errorf("handle swapoff %s failed, err: %s", req.Path, std) + } + if req.Operate == "delete" { + if req.WithRemove { + std, err := cmd.Execf("%s rm -rf %s", cmd.SudoHandleCmd(), req.Path) + if err != nil { + return fmt.Errorf("handle rm -rf %s failed, err: %s", req.Path, std) + } + } + if err := operateSwapWithFile("delete", req); err != nil { + return err + } + return nil + } + } + std1, err := cmd.Execf("%s dd if=/dev/zero of=%s bs=1024 count=%d", cmd.SudoHandleCmd(), req.Path, req.Size) + if err != nil { + _, _ = cmd.Execf("%s rm -rf %s", cmd.SudoHandleCmd(), req.Path) + return fmt.Errorf("handle dd path %s failed, err: %s", req.Path, std1) + } + std2, err := cmd.Execf("%s mkswap -f %s", cmd.SudoHandleCmd(), req.Path) + if err != nil { + _, _ = cmd.Execf("%s rm -rf %s", cmd.SudoHandleCmd(), req.Path) + return fmt.Errorf("handle dd path %s failed, err: %s", req.Path, std2) + } + std3, err := cmd.Execf("%s swapon %s", cmd.SudoHandleCmd(), req.Path) + if err != nil { + _, _ = cmd.Execf("%s swapoff %s", cmd.SudoHandleCmd(), req.Path) + _, _ = cmd.Execf("%s rm -rf %s", cmd.SudoHandleCmd(), req.Path) + return fmt.Errorf("handle dd path %s failed, err: %s", req.Path, std3) + } + if req.Operate == "create" { + if err := operateSwapWithFile("add", req); err != nil { + return err + } + return nil + } + return nil +} + func (u *DeviceService) LoadConf(name string) (string, error) { pathItem := "" switch name { @@ -291,3 +351,60 @@ func loadUser() string { } return strings.ReplaceAll(std, "\n", "") } + +func loadSwap() []dto.SwapHelper { + var data []dto.SwapHelper + std, err := cmd.Execf("%s swapon --show --summary", cmd.SudoHandleCmd()) + if err != nil { + return data + } + lines := strings.Split(std, "\n") + for index, line := range lines { + if index == 0 { + continue + } + parts := strings.Fields(line) + if len(parts) < 5 { + continue + } + sizeItem, _ := strconv.Atoi(parts[2]) + data = append(data, dto.SwapHelper{Path: parts[0], Size: uint64(sizeItem), Used: parts[3]}) + } + return data +} + +func operateSwapWithFile(operate string, req dto.SwapHelper) error { + if operate == "create" { + file, err := os.OpenFile(defaultFstab, os.O_WRONLY|os.O_TRUNC, 0640) + if err != nil { + return err + } + defer file.Close() + if _, err := file.WriteString(fmt.Sprintf("%s swap swap defaults 0 0 \n", req.Path)); err != nil { + return err + } + return nil + } + conf, err := os.ReadFile(defaultFstab) + if err != nil { + return fmt.Errorf("read file %s failed, err: %v", defaultFstab, err) + } + lines := strings.Split(string(conf), "\n") + newFile := "" + for _, line := range lines { + parts := strings.Fields(line) + if len(parts) == 6 && parts[0] == req.Path { + continue + } + newFile += line + "\n" + } + file, err := os.OpenFile(defaultHostPath, os.O_WRONLY|os.O_TRUNC, 0640) + if err != nil { + return err + } + defer file.Close() + write := bufio.NewWriter(file) + _, _ = write.WriteString(newFile) + write.Flush() + return nil +} diff --git a/backend/router/ro_toolbox.go b/backend/router/ro_toolbox.go index 64a58a63d548..acfe8f389607 100644 --- a/backend/router/ro_toolbox.go +++ b/backend/router/ro_toolbox.go @@ -21,6 +21,7 @@ func (s *ToolboxRouter) InitToolboxRouter(Router *gin.RouterGroup) { toolboxRouter.POST("/device/update/conf", baseApi.UpdateDeviceConf) toolboxRouter.POST("/device/update/host", baseApi.UpdateDeviceHost) toolboxRouter.POST("/device/update/passwd", baseApi.UpdateDevicPasswd) + toolboxRouter.POST("/device/update/swap", baseApi.UpdateDevicSwap) toolboxRouter.POST("/device/update/byconf", baseApi.UpdateDevicByFile) toolboxRouter.POST("/device/check/dns", baseApi.CheckDNS) toolboxRouter.POST("/device/conf", baseApi.LoadDeviceConf) diff --git a/frontend/src/api/interface/toolbox.ts b/frontend/src/api/interface/toolbox.ts index cb793d35e797..11164c99f474 100644 --- a/frontend/src/api/interface/toolbox.ts +++ b/frontend/src/api/interface/toolbox.ts @@ -7,6 +7,22 @@ export namespace Toolbox { user: string; timeZone: string; localTime: string; + + memoryTotal: number; + memoryAvailable: number; + memoryUsed: number; + swapMemoryTotal: number; + swapMemoryAvailable: number; + swapMemoryUsed: number; + + swapDetails: Array; + } + export interface SwapHelper { + operate: string; + path: string; + size: number; + used: number; + withRemove: boolean; } export interface HostHelper { ip: string; diff --git a/frontend/src/api/modules/toolbox.ts b/frontend/src/api/modules/toolbox.ts index ced950fd76e5..9c9f826398d8 100644 --- a/frontend/src/api/modules/toolbox.ts +++ b/frontend/src/api/modules/toolbox.ts @@ -2,6 +2,7 @@ import http from '@/api'; import { UpdateByFile } from '../interface'; import { Toolbox } from '../interface/toolbox'; import { Base64 } from 'js-base64'; +import { TimeoutEnum } from '@/enums/http-enum'; // device export const getDeviceBase = () => { @@ -19,6 +20,9 @@ export const updateDeviceHost = (param: Array) => { export const updateDevicePasswd = (user: string, passwd: string) => { return http.post(`/toolbox/device/update/passwd`, { user: user, passwd: Base64.encode(passwd) }); }; +export const updateDeviceSwap = (params: Toolbox.SwapHelper) => { + return http.post(`/toolbox/device/update/swap`, params, TimeoutEnum.T_5M); +}; export const updateDeviceByConf = (name: string, file: string) => { return http.post(`/toolbox/device/update/byconf`, { name: name, file: file }); }; diff --git a/frontend/src/lang/modules/zh.ts b/frontend/src/lang/modules/zh.ts index eb94800ffcb6..ada4660204c5 100644 --- a/frontend/src/lang/modules/zh.ts +++ b/frontend/src/lang/modules/zh.ts @@ -846,6 +846,9 @@ const message = { emptyTerminal: '暂无终端连接', }, toolbox: { + swap: { + saveHelper: '请先保存当前设置!', + }, device: { dnsHelper: '服务器地址域名解析', hostsHelper: '主机名解析', diff --git a/frontend/src/views/container/network/create/index.vue b/frontend/src/views/container/network/create/index.vue index e096aab9cf63..401295612a54 100644 --- a/frontend/src/views/container/network/create/index.vue +++ b/frontend/src/views/container/network/create/index.vue @@ -203,13 +203,51 @@ const rules = reactive({ name: [Rules.requiredInput], driver: [Rules.requiredSelect], subnet: [{ validator: checkCidr, trigger: 'blur' }], - gateway: [Rules.ip], + gateway: [{ validator: checkGateway, trigger: 'blur' }], scope: [{ validator: checkCidr, trigger: 'blur' }], subnetV6: [{ validator: checkFixedCidrV6, trigger: 'blur' }], - gatewayV6: [Rules.ipV6], + gatewayV6: [{ validator: checkGatewayV6, trigger: 'blur' }], scopeV6: [{ validator: checkFixedCidrV6, trigger: 'blur' }], }); +function checkGateway(rule: any, value: any, callback: any) { + if (value === '') { + callback(); + } + const reg = + /^(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])$/; + if (!reg.test(value) && value !== '') { + return callback(new Error(i18n.global.t('commons.rule.formatErr'))); + } + callback(); +} + +function checkGatewayV6(rule: any, value: any, callback: any) { + if (value === '') { + callback(); + } else { + const IPv4SegmentFormat = '(?:[0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])'; + const IPv4AddressFormat = `(${IPv4SegmentFormat}[.]){3}${IPv4SegmentFormat}`; + const IPv6SegmentFormat = '(?:[0-9a-fA-F]{1,4})'; + const IPv6AddressRegExp = new RegExp( + '^(' + + `(?:${IPv6SegmentFormat}:){7}(?:${IPv6SegmentFormat}|:)|` + + `(?:${IPv6SegmentFormat}:){6}(?:${IPv4AddressFormat}|:${IPv6SegmentFormat}|:)|` + + `(?:${IPv6SegmentFormat}:){5}(?::${IPv4AddressFormat}|(:${IPv6SegmentFormat}){1,2}|:)|` + + `(?:${IPv6SegmentFormat}:){4}(?:(:${IPv6SegmentFormat}){0,1}:${IPv4AddressFormat}|(:${IPv6SegmentFormat}){1,3}|:)|` + + `(?:${IPv6SegmentFormat}:){3}(?:(:${IPv6SegmentFormat}){0,2}:${IPv4AddressFormat}|(:${IPv6SegmentFormat}){1,4}|:)|` + + `(?:${IPv6SegmentFormat}:){2}(?:(:${IPv6SegmentFormat}){0,3}:${IPv4AddressFormat}|(:${IPv6SegmentFormat}){1,5}|:)|` + + `(?:${IPv6SegmentFormat}:){1}(?:(:${IPv6SegmentFormat}){0,4}:${IPv4AddressFormat}|(:${IPv6SegmentFormat}){1,6}|:)|` + + `(?::((?::${IPv6SegmentFormat}){0,5}:${IPv4AddressFormat}|(?::${IPv6SegmentFormat}){1,7}|:))` + + ')(%[0-9a-zA-Z-.:]{1,})?$', + ); + if (!IPv6AddressRegExp.test(value) && value !== '') { + return callback(new Error(i18n.global.t('commons.rule.formatErr'))); + } + callback(); + } +} + function checkCidr(rule: any, value: any, callback: any) { if (value === '') { callback(); diff --git a/frontend/src/views/home/status/index.vue b/frontend/src/views/home/status/index.vue index 5f60f90aae46..fe7be721aafc 100644 --- a/frontend/src/views/home/status/index.vue +++ b/frontend/src/views/home/status/index.vue @@ -71,7 +71,7 @@ {{ $t('home.free') }}: {{ formatNumber(currentInfo.swapMemoryAvailable / 1024 / 1024) }} MB - {{ $t('home.percent') }}: {{ formatNumber(100 - currentInfo.swapMemoryUsedPercent * 100) }}% + {{ $t('home.percent') }}: {{ formatNumber(currentInfo.swapMemoryUsedPercent * 100) }}% + + + + + + @@ -77,6 +87,7 @@ diff --git a/frontend/vite.config.ts b/frontend/vite.config.ts index bc39f4aab619..edbce645058f 100644 --- a/frontend/vite.config.ts +++ b/frontend/vite.config.ts @@ -84,7 +84,6 @@ export default defineConfig(({ mode }: ConfigEnv): UserConfig => { outDir: '../cmd/server/web', minify: 'esbuild', rollupOptions: { - external: ['codemirror'], output: { chunkFileNames: 'assets/js/[name]-[hash].js', entryFileNames: 'assets/js/[name]-[hash].js', From 9f5bebd3cdf023cb6c17f296ff70594b7b095937 Mon Sep 17 00:00:00 2001 From: ssongliu Date: Fri, 24 Nov 2023 15:31:34 +0800 Subject: [PATCH 2/6] =?UTF-8?q?fix:=20=E5=B7=A5=E5=85=B7=E7=AE=B1=E5=A2=9E?= =?UTF-8?q?=E5=8A=A0=20Swap=20=E6=94=AF=E6=8C=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- backend/app/dto/device.go | 5 ---- backend/app/service/device.go | 47 ++++++++--------------------------- 2 files changed, 10 insertions(+), 42 deletions(-) diff --git a/backend/app/dto/device.go b/backend/app/dto/device.go index b5b3616181c6..992f8b481d3d 100644 --- a/backend/app/dto/device.go +++ b/backend/app/dto/device.go @@ -9,9 +9,6 @@ type DeviceBaseInfo struct { Ntp string `json:"ntp"` User string `json:"user"` - MemoryTotal uint64 `json:"memoryTotal"` - MemoryAvailable uint64 `json:"memoryAvailable"` - MemoryUsed uint64 `json:"memoryUsed"` SwapMemoryTotal uint64 `json:"swapMemoryTotal"` SwapMemoryAvailable uint64 `json:"swapMemoryAvailable"` SwapMemoryUsed uint64 `json:"swapMemoryUsed"` @@ -29,8 +26,6 @@ type SwapHelper struct { Path string `json:"path" validate:"required"` Size uint64 `json:"size"` Used string `json:"used"` - - WithRemove bool `json:"withRemove"` } type TimeZoneOptions struct { diff --git a/backend/app/service/device.go b/backend/app/service/device.go index 668620d4c70d..9dc553b1be68 100644 --- a/backend/app/service/device.go +++ b/backend/app/service/device.go @@ -51,10 +51,6 @@ func (u *DeviceService) LoadBaseInfo() (dto.DeviceBaseInfo, error) { ntp, _ := settingRepo.Get(settingRepo.WithByKey("NtpSite")) baseInfo.Ntp = ntp.Value - memoryInfo, _ := mem.VirtualMemory() - baseInfo.MemoryTotal = memoryInfo.Total - baseInfo.MemoryAvailable = memoryInfo.Available - baseInfo.MemoryUsed = memoryInfo.Used swapInfo, _ := mem.SwapMemory() baseInfo.SwapMemoryTotal = swapInfo.Total baseInfo.SwapMemoryAvailable = swapInfo.Free @@ -191,41 +187,23 @@ func (u *DeviceService) UpdateSwap(req dto.SwapHelper) error { return fmt.Errorf("handle swapoff %s failed, err: %s", req.Path, std) } if req.Operate == "delete" { - if req.WithRemove { - std, err := cmd.Execf("%s rm -rf %s", cmd.SudoHandleCmd(), req.Path) - if err != nil { - return fmt.Errorf("handle rm -rf %s failed, err: %s", req.Path, std) - } - } - if err := operateSwapWithFile("delete", req); err != nil { - return err - } - return nil + return operateSwapWithFile(req) } } std1, err := cmd.Execf("%s dd if=/dev/zero of=%s bs=1024 count=%d", cmd.SudoHandleCmd(), req.Path, req.Size) if err != nil { - _, _ = cmd.Execf("%s rm -rf %s", cmd.SudoHandleCmd(), req.Path) return fmt.Errorf("handle dd path %s failed, err: %s", req.Path, std1) } std2, err := cmd.Execf("%s mkswap -f %s", cmd.SudoHandleCmd(), req.Path) if err != nil { - _, _ = cmd.Execf("%s rm -rf %s", cmd.SudoHandleCmd(), req.Path) return fmt.Errorf("handle dd path %s failed, err: %s", req.Path, std2) } std3, err := cmd.Execf("%s swapon %s", cmd.SudoHandleCmd(), req.Path) if err != nil { _, _ = cmd.Execf("%s swapoff %s", cmd.SudoHandleCmd(), req.Path) - _, _ = cmd.Execf("%s rm -rf %s", cmd.SudoHandleCmd(), req.Path) return fmt.Errorf("handle dd path %s failed, err: %s", req.Path, std3) } - if req.Operate == "create" { - if err := operateSwapWithFile("add", req); err != nil { - return err - } - return nil - } - return nil + return operateSwapWithFile(req) } func (u *DeviceService) LoadConf(name string) (string, error) { @@ -373,18 +351,7 @@ func loadSwap() []dto.SwapHelper { return data } -func operateSwapWithFile(operate string, req dto.SwapHelper) error { - if operate == "create" { - file, err := os.OpenFile(defaultFstab, os.O_WRONLY|os.O_TRUNC, 0640) - if err != nil { - return err - } - defer file.Close() - if _, err := file.WriteString(fmt.Sprintf("%s swap swap defaults 0 0 \n", req.Path)); err != nil { - return err - } - return nil - } +func operateSwapWithFile(req dto.SwapHelper) error { conf, err := os.ReadFile(defaultFstab) if err != nil { return fmt.Errorf("read file %s failed, err: %v", defaultFstab, err) @@ -392,13 +359,19 @@ func operateSwapWithFile(operate string, req dto.SwapHelper) error { lines := strings.Split(string(conf), "\n") newFile := "" for _, line := range lines { + if len(line) == 0 { + continue + } parts := strings.Fields(line) if len(parts) == 6 && parts[0] == req.Path { continue } newFile += line + "\n" } - file, err := os.OpenFile(defaultHostPath, os.O_WRONLY|os.O_TRUNC, 0640) + if req.Operate == "create" { + newFile += fmt.Sprintf("%s swap swap defaults 0 0\n", req.Path) + } + file, err := os.OpenFile(defaultFstab, os.O_WRONLY|os.O_TRUNC, 0640) if err != nil { return err } From 6472d9dcce94bc48121ac051a7477834f4d8fb36 Mon Sep 17 00:00:00 2001 From: ssongliu Date: Fri, 24 Nov 2023 15:32:50 +0800 Subject: [PATCH 3/6] =?UTF-8?q?feat:=20=E5=B7=A5=E5=85=B7=E7=AE=B1?= =?UTF-8?q?=E5=A2=9E=E5=8A=A0=20Swap=20=E7=AE=A1=E7=90=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- frontend/src/api/interface/toolbox.ts | 6 +- frontend/src/lang/modules/en.ts | 15 ++ frontend/src/lang/modules/tw.ts | 11 ++ frontend/src/lang/modules/zh.ts | 8 + frontend/src/views/toolbox/device/index.vue | 17 +- .../src/views/toolbox/device/swap/index.vue | 163 +++++++++--------- 6 files changed, 119 insertions(+), 101 deletions(-) diff --git a/frontend/src/api/interface/toolbox.ts b/frontend/src/api/interface/toolbox.ts index 11164c99f474..bbf89551ee72 100644 --- a/frontend/src/api/interface/toolbox.ts +++ b/frontend/src/api/interface/toolbox.ts @@ -8,9 +8,6 @@ export namespace Toolbox { timeZone: string; localTime: string; - memoryTotal: number; - memoryAvailable: number; - memoryUsed: number; swapMemoryTotal: number; swapMemoryAvailable: number; swapMemoryUsed: number; @@ -21,8 +18,7 @@ export namespace Toolbox { operate: string; path: string; size: number; - used: number; - withRemove: boolean; + used: string; } export interface HostHelper { ip: string; diff --git a/frontend/src/lang/modules/en.ts b/frontend/src/lang/modules/en.ts index 111dd2129585..42372d3b3b80 100644 --- a/frontend/src/lang/modules/en.ts +++ b/frontend/src/lang/modules/en.ts @@ -884,6 +884,21 @@ const message = { emptyTerminal: 'No terminal is currently connected', }, toolbox: { + swap: { + swap: 'Swap Partition', + swapHelper1: + 'The size of the swap should be 1 to 2 times the physical memory, adjustable based on specific requirements;', + swapHelper2: + 'Before creating a swap file, ensure that the system disk has sufficient available space, as the swap file size will occupy the corresponding disk space;', + swapHelper3: + 'Swap can help alleviate memory pressure, but it is only an alternative. Excessive reliance on swap may lead to a decrease in system performance. It is recommended to prioritize increasing memory or optimizing application memory usage;', + swapHelper4: 'It is advisable to regularly monitor the usage of swap to ensure normal system operation.', + swapDeleteHelper: + 'This operation will remove the Swap partition {0}. For system security reasons, the corresponding file will not be automatically deleted. If deletion is required, please proceed manually!', + saveHelper: 'Please save the current settings first!', + saveSwap: + 'Saving the current configuration will adjust the total usage of the swap partition to {0}. Do you want to continue?', + }, device: { dnsHelper: 'Server Address Domain Resolution', hostsHelper: 'Hostname Resolution', diff --git a/frontend/src/lang/modules/tw.ts b/frontend/src/lang/modules/tw.ts index b5a800bbc350..3d11baec23fe 100644 --- a/frontend/src/lang/modules/tw.ts +++ b/frontend/src/lang/modules/tw.ts @@ -845,6 +845,17 @@ const message = { emptyTerminal: '暫無終端連接', }, toolbox: { + swap: { + swap: 'Swap', + swapHelper1: 'Swap 的大小應該是物理內存的 1 到 2 倍,可根據具體情況進行調整;', + swapHelper2: '在創建 Swap 文件之前,請確保系統硬盤有足夠的可用空間,Swap 文件的大小將佔用相應的磁盤空間;', + swapHelper3: + 'Swap 可以幫助緩解內存壓力,但僅是一個備選項,過多依賴可能導致系統性能下降,建議優先考慮增加內存或者優化應用程序內存使用;', + swapHelper4: '建議定期監控 Swap 的使用情況,以確保系統正常運行。', + swapDeleteHelper: '此操作將移除 Swap 分區 {0},出於系統安全考慮,不會自動刪除該文件,如需刪除請手動操作!', + saveHelper: '請先保存當前設置!', + saveSwap: '保存當前配置將調整交換分區總佔用到 {0},是否繼續?', + }, device: { dnsHelper: '伺服器地址域名解析', hostsHelper: '主機名解析', diff --git a/frontend/src/lang/modules/zh.ts b/frontend/src/lang/modules/zh.ts index ada4660204c5..ed60f548c47b 100644 --- a/frontend/src/lang/modules/zh.ts +++ b/frontend/src/lang/modules/zh.ts @@ -847,7 +847,15 @@ const message = { }, toolbox: { swap: { + swap: 'Swap 分区', + swapHelper1: 'Swap 的大小应该是物理内存的 1 到 2 倍,可根据具体情况进行调整;', + swapHelper2: '在创建 Swap 文件之前,请确保系统硬盘有足够的可用空间,Swap 文件的大小将占用相应的磁盘空间;', + swapHelper3: + 'Swap 可以帮助缓解内存压力,但仅是一个备选项,过多依赖可能导致系统性能下降,建议优先考虑增加内存或者优化应用程序内存使用;', + swapHelper4: '建议定期监控 Swap 的使用情况,以确保系统正常运行。', + swapDeleteHelper: '此操作将移除 Swap 分区 {0},出于系统安全考虑,不会自动删除该文件,如需删除请手动操作!', saveHelper: '请先保存当前设置!', + saveSwap: '保存当前配置将调整 Swap 分区总占用到 {0},是否继续?', }, device: { dnsHelper: '服务器地址域名解析', diff --git a/frontend/src/views/toolbox/device/index.vue b/frontend/src/views/toolbox/device/index.vue index 499aa5a65385..f0274d8276d5 100644 --- a/frontend/src/views/toolbox/device/index.vue +++ b/frontend/src/views/toolbox/device/index.vue @@ -121,14 +121,6 @@ const form = reactive({ ntp: '', swapItem: '', - memoryTotal: '', - memoryAvailable: '', - memoryUsed: '', - swapMemoryTotal: '', - swapMemoryAvailable: '', - swapMemoryUsed: '', - - swapDetails: [], }); const onChangeTimeZone = () => { @@ -166,15 +158,8 @@ const search = async () => { form.hostItem = form.hosts ? i18n.global.t('toolbox.device.hostsHelper') : i18n.global.t('setting.unSetting'); form.swapItem = res.data.swapMemoryTotal - ? i18n.global.t('toolbox.device.dnsHelper') + ? computeSize(res.data.swapMemoryTotal) : i18n.global.t('setting.unSetting'); - form.memoryTotal = computeSize(res.data.memoryTotal); - form.memoryUsed = computeSize(res.data.memoryUsed); - form.memoryAvailable = computeSize(res.data.memoryAvailable); - form.swapMemoryTotal = computeSize(res.data.swapMemoryTotal); - form.swapMemoryUsed = computeSize(res.data.swapMemoryUsed); - form.swapMemoryAvailable = computeSize(res.data.swapMemoryAvailable); - form.swapDetails = res.data.swapDetails; }; onMounted(() => { diff --git a/frontend/src/views/toolbox/device/swap/index.vue b/frontend/src/views/toolbox/device/swap/index.vue index bb99cebdc462..5a5425be5c47 100644 --- a/frontend/src/views/toolbox/device/swap/index.vue +++ b/frontend/src/views/toolbox/device/swap/index.vue @@ -7,40 +7,18 @@ + + + - - - - - {{ form.memoryTotal }} - - - - - - {{ form.memoryUsed }} - - - - - - {{ form.memoryAvailable }} - - - @@ -73,13 +51,21 @@ @@ -135,12 +121,9 @@ const isCreate = ref(); const form = reactive({ swapItem: '', - memoryTotal: 0, - memoryAvailable: 0, - memoryUsed: 0, - swapMemoryTotal: 0, - swapMemoryAvailable: 0, - swapMemoryUsed: 0, + swapMemoryTotal: '', + swapMemoryAvailable: '', + swapMemoryUsed: '', swapDetails: [], }); @@ -167,36 +150,14 @@ const handleAdd = () => { isCreate.value = true; form.swapDetails.push(item); }; -const handleDelete = async (row: any) => { - let params = { - operate: 'delete', - path: row.path, - size: 0, - used: '0', - withRemove: true, - }; - loading.value = true; - await updateDeviceSwap(params) - .then(() => { - loading.value = false; - MsgSuccess(i18n.global.t('commons.msg.operationSuccess')); - search(); - }) - .catch(() => { - loading.value = false; - }); -}; const search = async () => { isCreate.value = false; const res = await getDeviceBase(); - form.memoryTotal = computeSize(res.data.memoryTotal); - form.memoryUsed = computeSize(res.data.memoryUsed); - form.memoryAvailable = computeSize(res.data.memoryAvailable); form.swapMemoryTotal = computeSize(res.data.swapMemoryTotal); form.swapMemoryUsed = computeSize(res.data.swapMemoryUsed); form.swapMemoryAvailable = computeSize(res.data.swapMemoryAvailable); - form.swapDetails = res.data.swapDetails; + form.swapDetails = res.data.swapDetails || []; for (const item of form.swapDetails) { item.isCreate = false; item.isEdit = false; @@ -204,24 +165,66 @@ const search = async () => { } }; +const handleDelete = async (row: any) => { + ElMessageBox.confirm( + i18n.global.t('toolbox.swap.swapDeleteHelper', [row.path]), + i18n.global.t('commons.button.delete'), + { + confirmButtonText: i18n.global.t('commons.button.confirm'), + cancelButtonText: i18n.global.t('commons.button.cancel'), + type: 'info', + }, + ).then(async () => { + let params = { + operate: 'delete', + path: row.path, + size: 0, + used: '0', + }; + loading.value = true; + await updateDeviceSwap(params) + .then(() => { + loading.value = false; + MsgSuccess(i18n.global.t('commons.msg.operationSuccess')); + search(); + }) + .catch(() => { + loading.value = false; + }); + }); +}; + const onSave = async (row) => { - let params = { - operate: row.isCreate ? 'create' : 'update', - path: row.path, - size: row.size * 1024, - used: '0', - withRemove: false, - }; - loading.value = true; - await updateDeviceSwap(params) - .then(() => { - loading.value = false; - MsgSuccess(i18n.global.t('commons.msg.operationSuccess')); - search(); - }) - .catch(() => { - loading.value = false; - }); + let itemSize = 0; + for (const item of form.swapDetails) { + itemSize += item.size; + } + ElMessageBox.confirm( + i18n.global.t('toolbox.swap.saveSwap', [itemSize + ' M']), + i18n.global.t('commons.button.save'), + { + confirmButtonText: i18n.global.t('commons.button.confirm'), + cancelButtonText: i18n.global.t('commons.button.cancel'), + type: 'info', + }, + ).then(async () => { + let params = { + operate: row.isCreate ? 'create' : 'update', + path: row.path, + size: row.size * 1024, + used: '0', + }; + loading.value = true; + await updateDeviceSwap(params) + .then(() => { + loading.value = false; + MsgSuccess(i18n.global.t('commons.msg.operationSuccess')); + search(); + }) + .catch(() => { + loading.value = false; + }); + }); }; const handleClose = () => { From 4fa86caf5d5f6ca5f7fc43def42ed7883583d73c Mon Sep 17 00:00:00 2001 From: ssongliu Date: Fri, 24 Nov 2023 15:58:10 +0800 Subject: [PATCH 4/6] =?UTF-8?q?feat:=20=E4=BF=AE=E6=94=B9=E6=8E=A5?= =?UTF-8?q?=E5=8F=A3=E6=96=87=E6=A1=A3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- backend/app/api/v1/device.go | 8 ++-- backend/router/ro_toolbox.go | 6 +-- cmd/server/docs/docs.go | 87 ++++++++++++++++++++++++++++++++++++ cmd/server/docs/swagger.json | 87 ++++++++++++++++++++++++++++++++++++ cmd/server/docs/swagger.yaml | 58 ++++++++++++++++++++++++ 5 files changed, 239 insertions(+), 7 deletions(-) diff --git a/backend/app/api/v1/device.go b/backend/app/api/v1/device.go index fc9ee3ca49f4..8945b8e06f85 100644 --- a/backend/app/api/v1/device.go +++ b/backend/app/api/v1/device.go @@ -73,7 +73,7 @@ func (b *BaseApi) LoadDeviceConf(c *gin.Context) { // @Success 200 // @Security ApiKeyAuth // @Router /toolbox/device/update/byconf [post] -func (b *BaseApi) UpdateDevicByFile(c *gin.Context) { +func (b *BaseApi) UpdateDeviceByFile(c *gin.Context) { var req dto.UpdateByNameAndFile if err := helper.CheckBindAndValidate(&req, c); err != nil { return @@ -138,7 +138,7 @@ func (b *BaseApi) UpdateDeviceHost(c *gin.Context) { // @Success 200 // @Security ApiKeyAuth // @Router /toolbox/device/update/passwd [post] -func (b *BaseApi) UpdateDevicPasswd(c *gin.Context) { +func (b *BaseApi) UpdateDevicePasswd(c *gin.Context) { var req dto.ChangePasswd if err := helper.CheckBindAndValidate(&req, c); err != nil { return @@ -167,8 +167,8 @@ func (b *BaseApi) UpdateDevicPasswd(c *gin.Context) { // @Success 200 // @Security ApiKeyAuth // @Router /toolbox/device/update/swap [post] -// @x-panel-log {"bodyKeys":["opreate","path"],"paramKeys":[],"BeforeFunctions":[],"formatZH":"[opreate] 主机 swap [path]","formatEN":"[opreate] device swap [path]"} -func (b *BaseApi) UpdateDevicSwap(c *gin.Context) { +// @x-panel-log {"bodyKeys":["operate","path"],"paramKeys":[],"BeforeFunctions":[],"formatZH":"[operate] 主机 swap [path]","formatEN":"[operate] device swap [path]"} +func (b *BaseApi) UpdateDeviceSwap(c *gin.Context) { var req dto.SwapHelper if err := helper.CheckBindAndValidate(&req, c); err != nil { return diff --git a/backend/router/ro_toolbox.go b/backend/router/ro_toolbox.go index acfe8f389607..fb86e8ca0236 100644 --- a/backend/router/ro_toolbox.go +++ b/backend/router/ro_toolbox.go @@ -20,9 +20,9 @@ func (s *ToolboxRouter) InitToolboxRouter(Router *gin.RouterGroup) { toolboxRouter.GET("/device/zone/options", baseApi.LoadTimeOption) toolboxRouter.POST("/device/update/conf", baseApi.UpdateDeviceConf) toolboxRouter.POST("/device/update/host", baseApi.UpdateDeviceHost) - toolboxRouter.POST("/device/update/passwd", baseApi.UpdateDevicPasswd) - toolboxRouter.POST("/device/update/swap", baseApi.UpdateDevicSwap) - toolboxRouter.POST("/device/update/byconf", baseApi.UpdateDevicByFile) + toolboxRouter.POST("/device/update/passwd", baseApi.UpdateDevicePasswd) + toolboxRouter.POST("/device/update/swap", baseApi.UpdateDeviceSwap) + toolboxRouter.POST("/device/update/byconf", baseApi.UpdateDeviceByFile) toolboxRouter.POST("/device/check/dns", baseApi.CheckDNS) toolboxRouter.POST("/device/conf", baseApi.LoadDeviceConf) diff --git a/cmd/server/docs/docs.go b/cmd/server/docs/docs.go index 2e53c8099847..063f2edeacfb 100644 --- a/cmd/server/docs/docs.go +++ b/cmd/server/docs/docs.go @@ -10331,6 +10331,49 @@ const docTemplate = `{ } } }, + "/toolbox/device/update/swap": { + "post": { + "security": [ + { + "ApiKeyAuth": [] + } + ], + "description": "修改系统 Swap", + "consumes": [ + "application/json" + ], + "tags": [ + "Device" + ], + "summary": "Update device swap", + "parameters": [ + { + "description": "request", + "name": "request", + "in": "body", + "required": true, + "schema": { + "$ref": "#/definitions/dto.SwapHelper" + } + } + ], + "responses": { + "200": { + "description": "OK" + } + }, + "x-panel-log": { + "BeforeFunctions": [], + "bodyKeys": [ + "operate", + "path" + ], + "formatEN": "[operate] device swap [path]", + "formatZH": "[operate] 主机 swap [path]", + "paramKeys": [] + } + } + }, "/toolbox/device/zone/options": { "get": { "security": [ @@ -13748,6 +13791,9 @@ const docTemplate = `{ "openStdin": { "type": "boolean" }, + "privileged": { + "type": "boolean" + }, "publishAllPorts": { "type": "boolean" }, @@ -14519,6 +14565,21 @@ const docTemplate = `{ "ntp": { "type": "string" }, + "swapDetails": { + "type": "array", + "items": { + "$ref": "#/definitions/dto.SwapHelper" + } + }, + "swapMemoryAvailable": { + "type": "integer" + }, + "swapMemoryTotal": { + "type": "integer" + }, + "swapMemoryUsed": { + "type": "integer" + }, "timeZone": { "type": "string" }, @@ -16653,6 +16714,32 @@ const docTemplate = `{ } } }, + "dto.SwapHelper": { + "type": "object", + "required": [ + "operate", + "path" + ], + "properties": { + "operate": { + "type": "string", + "enum": [ + "create", + "delete", + "update" + ] + }, + "path": { + "type": "string" + }, + "size": { + "type": "integer" + }, + "used": { + "type": "string" + } + } + }, "dto.TreeChild": { "type": "object", "properties": { diff --git a/cmd/server/docs/swagger.json b/cmd/server/docs/swagger.json index 73e057a8a02b..41fa03ad92e6 100644 --- a/cmd/server/docs/swagger.json +++ b/cmd/server/docs/swagger.json @@ -10324,6 +10324,49 @@ } } }, + "/toolbox/device/update/swap": { + "post": { + "security": [ + { + "ApiKeyAuth": [] + } + ], + "description": "修改系统 Swap", + "consumes": [ + "application/json" + ], + "tags": [ + "Device" + ], + "summary": "Update device swap", + "parameters": [ + { + "description": "request", + "name": "request", + "in": "body", + "required": true, + "schema": { + "$ref": "#/definitions/dto.SwapHelper" + } + } + ], + "responses": { + "200": { + "description": "OK" + } + }, + "x-panel-log": { + "BeforeFunctions": [], + "bodyKeys": [ + "operate", + "path" + ], + "formatEN": "[operate] device swap [path]", + "formatZH": "[operate] 主机 swap [path]", + "paramKeys": [] + } + } + }, "/toolbox/device/zone/options": { "get": { "security": [ @@ -13741,6 +13784,9 @@ "openStdin": { "type": "boolean" }, + "privileged": { + "type": "boolean" + }, "publishAllPorts": { "type": "boolean" }, @@ -14512,6 +14558,21 @@ "ntp": { "type": "string" }, + "swapDetails": { + "type": "array", + "items": { + "$ref": "#/definitions/dto.SwapHelper" + } + }, + "swapMemoryAvailable": { + "type": "integer" + }, + "swapMemoryTotal": { + "type": "integer" + }, + "swapMemoryUsed": { + "type": "integer" + }, "timeZone": { "type": "string" }, @@ -16646,6 +16707,32 @@ } } }, + "dto.SwapHelper": { + "type": "object", + "required": [ + "operate", + "path" + ], + "properties": { + "operate": { + "type": "string", + "enum": [ + "create", + "delete", + "update" + ] + }, + "path": { + "type": "string" + }, + "size": { + "type": "integer" + }, + "used": { + "type": "string" + } + } + }, "dto.TreeChild": { "type": "object", "properties": { diff --git a/cmd/server/docs/swagger.yaml b/cmd/server/docs/swagger.yaml index 0b2304c57549..cd19f432ffe5 100644 --- a/cmd/server/docs/swagger.yaml +++ b/cmd/server/docs/swagger.yaml @@ -433,6 +433,8 @@ definitions: type: string openStdin: type: boolean + privileged: + type: boolean publishAllPorts: type: boolean restartPolicy: @@ -954,6 +956,16 @@ definitions: type: string ntp: type: string + swapDetails: + items: + $ref: '#/definitions/dto.SwapHelper' + type: array + swapMemoryAvailable: + type: integer + swapMemoryTotal: + type: integer + swapMemoryUsed: + type: integer timeZone: type: string user: @@ -2395,6 +2407,24 @@ definitions: required: - id type: object + dto.SwapHelper: + properties: + operate: + enum: + - create + - delete + - update + type: string + path: + type: string + size: + type: integer + used: + type: string + required: + - operate + - path + type: object dto.TreeChild: properties: id: @@ -11107,6 +11137,34 @@ paths: summary: Update device passwd tags: - Device + /toolbox/device/update/swap: + post: + consumes: + - application/json + description: 修改系统 Swap + parameters: + - description: request + in: body + name: request + required: true + schema: + $ref: '#/definitions/dto.SwapHelper' + responses: + "200": + description: OK + security: + - ApiKeyAuth: [] + summary: Update device swap + tags: + - Device + x-panel-log: + BeforeFunctions: [] + bodyKeys: + - operate + - path + formatEN: '[operate] device swap [path]' + formatZH: '[operate] 主机 swap [path]' + paramKeys: [] /toolbox/device/zone/options: get: consumes: From a53b2b890ef959a0af944d7dc3aea2d94610016b Mon Sep 17 00:00:00 2001 From: ssongliu Date: Mon, 27 Nov 2023 11:53:23 +0800 Subject: [PATCH 5/6] =?UTF-8?q?=20fix:=20=E9=80=BB=E8=BE=91=E8=B0=83?= =?UTF-8?q?=E6=95=B4?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- backend/app/dto/device.go | 9 +++++---- backend/app/service/device.go | 33 +++++++++++++++++++++++---------- 2 files changed, 28 insertions(+), 14 deletions(-) diff --git a/backend/app/dto/device.go b/backend/app/dto/device.go index 992f8b481d3d..7aa283bd9d4a 100644 --- a/backend/app/dto/device.go +++ b/backend/app/dto/device.go @@ -22,10 +22,11 @@ type HostHelper struct { } type SwapHelper struct { - Operate string `json:"operate" validate:"required,oneof=create delete update"` - Path string `json:"path" validate:"required"` - Size uint64 `json:"size"` - Used string `json:"used"` + Path string `json:"path" validate:"required"` + Size uint64 `json:"size"` + Used string `json:"used"` + + IsNew bool `json:"isNew"` } type TimeZoneOptions struct { diff --git a/backend/app/service/device.go b/backend/app/service/device.go index 9dc553b1be68..0d4b00c73f1e 100644 --- a/backend/app/service/device.go +++ b/backend/app/service/device.go @@ -6,6 +6,7 @@ import ( "fmt" "net" "os" + "path" "strconv" "strings" "time" @@ -116,11 +117,20 @@ func (u *DeviceService) Update(key, value string) error { if err != nil { return errors.New(std) } - case "LocalTime": - if err := settingRepo.Update("NtpSite", value); err != nil { - return err + case "Ntp", "LocalTime": + ntpValue := value + if key == "LocalTime" { + ntpItem, err := settingRepo.Get(settingRepo.WithByKey("NtpSite")) + if err != nil { + return err + } + ntpValue = ntpItem.Value + } else { + if err := settingRepo.Update("NtpSite", ntpValue); err != nil { + return err + } } - ntime, err := ntp.GetRemoteTime(value) + ntime, err := ntp.GetRemoteTime(ntpValue) if err != nil { return err } @@ -181,14 +191,17 @@ func (u *DeviceService) UpdatePasswd(req dto.ChangePasswd) error { } func (u *DeviceService) UpdateSwap(req dto.SwapHelper) error { - if req.Operate != "create" { + if !req.IsNew { std, err := cmd.Execf("%s swapoff %s", cmd.SudoHandleCmd(), req.Path) if err != nil { return fmt.Errorf("handle swapoff %s failed, err: %s", req.Path, std) } - if req.Operate == "delete" { - return operateSwapWithFile(req) + } + if req.Size == 0 { + if req.Path == path.Join(global.CONF.System.BaseDir, ".1panel_swap") { + _ = os.Remove(path.Join(global.CONF.System.BaseDir, ".1panel_swap")) } + return operateSwapWithFile(true, req) } std1, err := cmd.Execf("%s dd if=/dev/zero of=%s bs=1024 count=%d", cmd.SudoHandleCmd(), req.Path, req.Size) if err != nil { @@ -203,7 +216,7 @@ func (u *DeviceService) UpdateSwap(req dto.SwapHelper) error { _, _ = cmd.Execf("%s swapoff %s", cmd.SudoHandleCmd(), req.Path) return fmt.Errorf("handle dd path %s failed, err: %s", req.Path, std3) } - return operateSwapWithFile(req) + return operateSwapWithFile(false, req) } func (u *DeviceService) LoadConf(name string) (string, error) { @@ -351,7 +364,7 @@ func loadSwap() []dto.SwapHelper { return data } -func operateSwapWithFile(req dto.SwapHelper) error { +func operateSwapWithFile(delete bool, req dto.SwapHelper) error { conf, err := os.ReadFile(defaultFstab) if err != nil { return fmt.Errorf("read file %s failed, err: %v", defaultFstab, err) @@ -368,7 +381,7 @@ func operateSwapWithFile(req dto.SwapHelper) error { } newFile += line + "\n" } - if req.Operate == "create" { + if !delete { newFile += fmt.Sprintf("%s swap swap defaults 0 0\n", req.Path) } file, err := os.OpenFile(defaultFstab, os.O_WRONLY|os.O_TRUNC, 0640) From 6bd6c211e8c60d789aae943e0ba4f97025e5ed06 Mon Sep 17 00:00:00 2001 From: ssongliu Date: Mon, 27 Nov 2023 11:54:22 +0800 Subject: [PATCH 6/6] =?UTF-8?q?=20fix:=20=E9=80=BB=E8=BE=91=E8=B0=83?= =?UTF-8?q?=E6=95=B4?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- frontend/src/api/interface/toolbox.ts | 3 +- frontend/src/lang/modules/en.ts | 6 +- frontend/src/lang/modules/tw.ts | 8 +- frontend/src/lang/modules/zh.ts | 8 +- frontend/src/utils/util.ts | 23 ++- frontend/src/views/toolbox/device/index.vue | 40 +++-- .../device/{local-time => ntp}/index.vue | 13 +- .../src/views/toolbox/device/swap/index.vue | 144 +++++++----------- 8 files changed, 123 insertions(+), 122 deletions(-) rename frontend/src/views/toolbox/device/{local-time => ntp}/index.vue (87%) diff --git a/frontend/src/api/interface/toolbox.ts b/frontend/src/api/interface/toolbox.ts index bbf89551ee72..d27497445b43 100644 --- a/frontend/src/api/interface/toolbox.ts +++ b/frontend/src/api/interface/toolbox.ts @@ -15,10 +15,11 @@ export namespace Toolbox { swapDetails: Array; } export interface SwapHelper { - operate: string; path: string; size: number; used: string; + + isNew: boolean; } export interface HostHelper { ip: string; diff --git a/frontend/src/lang/modules/en.ts b/frontend/src/lang/modules/en.ts index 42372d3b3b80..865dc7890ac2 100644 --- a/frontend/src/lang/modules/en.ts +++ b/frontend/src/lang/modules/en.ts @@ -897,7 +897,9 @@ const message = { 'This operation will remove the Swap partition {0}. For system security reasons, the corresponding file will not be automatically deleted. If deletion is required, please proceed manually!', saveHelper: 'Please save the current settings first!', saveSwap: - 'Saving the current configuration will adjust the total usage of the swap partition to {0}. Do you want to continue?', + 'Saving the current configuration will adjust the Swap partition {0} size to {1}. Do you want to continue?', + saveSwapHelper: 'The minimum partition size is 40 KB. Please modify and try again!', + swapOff: 'The minimum partition size is 40 KB. Setting it to 0 will disable the Swap partition.', }, device: { dnsHelper: 'Server Address Domain Resolution', @@ -905,7 +907,7 @@ const message = { hosts: 'Domain', toolbox: 'Toolbox', hostname: 'Hostname', - passwd: 'Host Password', + passwd: 'System Password', passwdHelper: 'Input characters cannot include $ and &', timeZone: 'System Time Zone', localTime: 'Server Time', diff --git a/frontend/src/lang/modules/tw.ts b/frontend/src/lang/modules/tw.ts index 3d11baec23fe..20c452c127f3 100644 --- a/frontend/src/lang/modules/tw.ts +++ b/frontend/src/lang/modules/tw.ts @@ -854,7 +854,9 @@ const message = { swapHelper4: '建議定期監控 Swap 的使用情況,以確保系統正常運行。', swapDeleteHelper: '此操作將移除 Swap 分區 {0},出於系統安全考慮,不會自動刪除該文件,如需刪除請手動操作!', saveHelper: '請先保存當前設置!', - saveSwap: '保存當前配置將調整交換分區總佔用到 {0},是否繼續?', + saveSwap: '儲存當前配置將調整 Swap 分區 {0} 大小到 {1},是否繼續?', + saveSwapHelper: '分區大小最小值為 40 KB,請修改後重試!', + swapOff: '分區大小最小值為 40 KB,設置為 0 則關閉 Swap 分區。', }, device: { dnsHelper: '伺服器地址域名解析', @@ -862,7 +864,7 @@ const message = { hosts: '域名', toolbox: '工具箱', hostname: '主機名', - passwd: '主機密碼', + passwd: '系統密碼', passwdHelper: '輸入的字符不能包含 $ 和 &', timeZone: '系統時區', localTime: '伺服器時間', @@ -874,6 +876,7 @@ const message = { ntpALi: '阿里', ntpGoogle: '谷歌', syncSite: 'NTP 伺服器', + syncSiteHelper: '該操作將使用 {0} 作為源進行系統時間同步,是否繼續?', hostnameHelper: '主機名修改依賴於 hostnamectl 命令,如未安裝可能導致修改失敗', userHelper: '用戶名依賴於 whoami 命令獲取,如未安裝可能導致獲取失敗。', passwordHelper: '密碼修改依賴於 chpasswd 命令,如未安裝可能導致修改失敗', @@ -1101,7 +1104,6 @@ const message = { systemIP: '服務器地址', systemIPWarning: '當前未設置服務器地址,請先在面板設置中設置!', defaultNetwork: '默認網卡', - syncSiteHelper: '該操作將使用 {0} 作為源進行系統時間同步,是否繼續?', changePassword: '密碼修改', oldPassword: '原密碼', newPassword: '新密碼', diff --git a/frontend/src/lang/modules/zh.ts b/frontend/src/lang/modules/zh.ts index ed60f548c47b..908500d4390f 100644 --- a/frontend/src/lang/modules/zh.ts +++ b/frontend/src/lang/modules/zh.ts @@ -855,7 +855,9 @@ const message = { swapHelper4: '建议定期监控 Swap 的使用情况,以确保系统正常运行。', swapDeleteHelper: '此操作将移除 Swap 分区 {0},出于系统安全考虑,不会自动删除该文件,如需删除请手动操作!', saveHelper: '请先保存当前设置!', - saveSwap: '保存当前配置将调整 Swap 分区总占用到 {0},是否继续?', + saveSwap: '保存当前配置将调整 Swap 分区 {0} 大小到 {1},是否继续?', + saveSwapHelper: '分区大小最小值为 40 KB,请修改后重试!', + swapOff: '分区大小最小值为 40 KB,设置成 0 则关闭 Swap 分区。', }, device: { dnsHelper: '服务器地址域名解析', @@ -863,7 +865,7 @@ const message = { hosts: '域名', toolbox: '工具箱', hostname: '主机名', - passwd: '主机密码', + passwd: '系统密码', passwdHelper: '输入字符不能包含 $ 和 &', timeZone: '系统时区', localTime: '服务器时间', @@ -875,6 +877,7 @@ const message = { ntpALi: '阿里', ntpGoogle: '谷歌', syncSite: 'NTP 服务器', + syncSiteHelper: '该操作将使用 {0} 作为源进行系统时间同步,是否继续?', hostnameHelper: '主机名修改依赖于 hostnamectl 命令,如未安装可能导致修改失败', userHelper: '用户名依赖于 whoami 命令获取,如未安装可能导致获取失败。', passwordHelper: '密码修改依赖于 chpasswd 命令,如未安装可能导致修改失败', @@ -1102,7 +1105,6 @@ const message = { systemIP: '服务器地址', systemIPWarning: '当前未设置服务器地址,请先在面板设置中设置!', defaultNetwork: '默认网卡', - syncSiteHelper: '该操作将使用 {0} 作为源进行系统时间同步,是否继续?', changePassword: '密码修改', oldPassword: '原密码', newPassword: '新密码', diff --git a/frontend/src/utils/util.ts b/frontend/src/utils/util.ts index 5053d3cacf0a..48190ba8c9ae 100644 --- a/frontend/src/utils/util.ts +++ b/frontend/src/utils/util.ts @@ -134,10 +134,25 @@ export function loadZero(i: number) { export function computeSize(size: number): string { const num = 1024.0; if (size < num) return size + ' B'; - if (size < Math.pow(num, 2)) return (size / num).toFixed(2) + ' KB'; - if (size < Math.pow(num, 3)) return (size / Math.pow(num, 2)).toFixed(2) + ' MB'; - if (size < Math.pow(num, 4)) return (size / Math.pow(num, 3)).toFixed(2) + ' GB'; - return (size / Math.pow(num, 4)).toFixed(2) + ' TB'; + if (size < Math.pow(num, 2)) return formattedNumber((size / num).toFixed(2)) + ' KB'; + if (size < Math.pow(num, 3)) return formattedNumber((size / Math.pow(num, 2)).toFixed(2)) + ' MB'; + if (size < Math.pow(num, 4)) return formattedNumber((size / Math.pow(num, 3)).toFixed(2)) + ' GB'; + return formattedNumber((size / Math.pow(num, 4)).toFixed(2)) + ' TB'; +} + +export function splitSize(size: number): any { + const num = 1024.0; + if (size < num) return { size: Number(size), unit: 'B' }; + if (size < Math.pow(num, 2)) return { size: formattedNumber((size / num).toFixed(2)), unit: 'KB' }; + if (size < Math.pow(num, 3)) + return { size: formattedNumber((size / Number(Math.pow(num, 2).toFixed(2))).toFixed(2)), unit: 'MB' }; + if (size < Math.pow(num, 4)) + return { size: formattedNumber((size / Number(Math.pow(num, 3))).toFixed(2)), unit: 'GB' }; + return { size: formattedNumber((size / Number(Math.pow(num, 4))).toFixed(2)), unit: 'TB' }; +} + +export function formattedNumber(num: string) { + return num.endsWith('.00') ? Number(num.slice(0, -3)) : Number(num); } export function computeSizeFromMB(size: number): string { diff --git a/frontend/src/views/toolbox/device/index.vue b/frontend/src/views/toolbox/device/index.vue index f0274d8276d5..db1a5a3bac31 100644 --- a/frontend/src/views/toolbox/device/index.vue +++ b/frontend/src/views/toolbox/device/index.vue @@ -43,7 +43,7 @@ - + + + + + + @@ -49,21 +45,18 @@ import { updateDevice } from '@/api/modules/toolbox'; const emit = defineEmits<{ (e: 'search'): void }>(); interface DialogProps { - localTime: string; ntpSite: string; } const drawerVisible = ref(); const loading = ref(); const form = reactive({ - localTime: '', ntpSite: '', }); const formRef = ref(); const acceptParams = (params: DialogProps): void => { - form.localTime = params.localTime; form.ntpSite = params.ntpSite; drawerVisible.value = true; }; @@ -82,7 +75,7 @@ const onSyncTime = async (formEl: FormInstance | undefined) => { }, ).then(async () => { loading.value = true; - await updateDevice('LocalTime', form.ntpSite) + await updateDevice('Ntp', form.ntpSite) .then(() => { loading.value = false; emit('search'); diff --git a/frontend/src/views/toolbox/device/swap/index.vue b/frontend/src/views/toolbox/device/swap/index.vue index 5a5425be5c47..7725f7331b53 100644 --- a/frontend/src/views/toolbox/device/swap/index.vue +++ b/frontend/src/views/toolbox/device/swap/index.vue @@ -49,56 +49,38 @@ - + - + - - {{ $t('commons.button.add') }} - + {{ $t('toolbox.swap.swapOff') }}