feat: 添加对本地文件的操作接口
This commit is contained in:
@@ -16,7 +16,7 @@ type FileListRow struct {
|
|||||||
LinkCount int64 `json:"linkCount"` // 硬链接数目
|
LinkCount int64 `json:"linkCount"` // 硬链接数目
|
||||||
Owner string `json:"owner"` // 所属用户
|
Owner string `json:"owner"` // 所属用户
|
||||||
Group string `json:"group"` // 所属组
|
Group string `json:"group"` // 所属组
|
||||||
Size string `json:"size"` // 文件的大小
|
Size int64 `json:"size"` // 文件的大小
|
||||||
ModifiedTime int64 `json:"modifiedTime"` // 最后修改时间,单位为秒
|
ModifiedTime int64 `json:"modifiedTime"` // 最后修改时间,单位为秒
|
||||||
FileName string `json:"fileName"` // 文件的名称
|
FileName string `json:"fileName"` // 文件的名称
|
||||||
}
|
}
|
||||||
@@ -34,10 +34,10 @@ func FileList(sshClient *ConnSSH, path, search string) ([]FileListRow, error) {
|
|||||||
if search != "" {
|
if search != "" {
|
||||||
searchStr = search + searchStr
|
searchStr = search + searchStr
|
||||||
}
|
}
|
||||||
// cd /var/log && find. -maxdepth 1 -name'mme*' -exec ls -lthd --time-style=+%s {} +
|
// cd /var/log && find. -maxdepth 1 -name'mme*' -exec ls -ltd --time-style=+%s {} +
|
||||||
cmdStr := fmt.Sprintf("cd %s && find . -maxdepth 1 -name '%s' -exec ls -lthd --time-style=+%%s {} +", path, searchStr)
|
cmdStr := fmt.Sprintf("cd %s && find . -maxdepth 1 -name '%s' -exec ls -ltd --time-style=+%%s {} +", path, searchStr)
|
||||||
// cd /var/log && ls -lthd --time-style=+%s mme*
|
// cd /var/log && ls -ltd --time-style=+%s mme*
|
||||||
// cmdStr := fmt.Sprintf("cd %s && ls -lthd --time-style=+%%s %s", path, searchStr)
|
// cmdStr := fmt.Sprintf("cd %s && ls -ltd --time-style=+%%s %s", path, searchStr)
|
||||||
|
|
||||||
// 是否远程客户端读取
|
// 是否远程客户端读取
|
||||||
if sshClient == nil {
|
if sshClient == nil {
|
||||||
@@ -94,7 +94,7 @@ func FileList(sshClient *ConnSSH, path, search string) ([]FileListRow, error) {
|
|||||||
LinkCount: parse.Number(fields[1]),
|
LinkCount: parse.Number(fields[1]),
|
||||||
Owner: fields[2],
|
Owner: fields[2],
|
||||||
Group: fields[3],
|
Group: fields[3],
|
||||||
Size: fields[4],
|
Size: parse.Number(fields[4]),
|
||||||
ModifiedTime: parse.Number(fields[5]),
|
ModifiedTime: parse.Number(fields[5]),
|
||||||
FileName: fileName,
|
FileName: fileName,
|
||||||
})
|
})
|
||||||
|
|||||||
78
src/framework/utils/file/files.go
Normal file
78
src/framework/utils/file/files.go
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
package file
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
)
|
||||||
|
|
||||||
|
// FileListRow 文件列表行数据
|
||||||
|
type FileListRow struct {
|
||||||
|
FileType string `json:"fileType"` // 文件类型 dir, file, symlink
|
||||||
|
FileMode string `json:"fileMode"` // 文件的权限
|
||||||
|
LinkCount int64 `json:"linkCount"` // 硬链接数目
|
||||||
|
Owner string `json:"owner"` // 所属用户
|
||||||
|
Group string `json:"group"` // 所属组
|
||||||
|
Size int64 `json:"size"` // 文件的大小
|
||||||
|
ModifiedTime int64 `json:"modifiedTime"` // 最后修改时间,单位为秒
|
||||||
|
FileName string `json:"fileName"` // 文件的名称
|
||||||
|
}
|
||||||
|
|
||||||
|
// 文件列表
|
||||||
|
// search 文件名后模糊*
|
||||||
|
//
|
||||||
|
// return 行记录,异常
|
||||||
|
func FileList(path, search string) ([]FileListRow, error) {
|
||||||
|
var rows []FileListRow
|
||||||
|
|
||||||
|
// 构建搜索模式
|
||||||
|
pattern := "*"
|
||||||
|
if search != "" {
|
||||||
|
pattern = search + pattern
|
||||||
|
}
|
||||||
|
|
||||||
|
// 读取目录内容
|
||||||
|
entries, err := os.ReadDir(path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// 遍历目录项
|
||||||
|
for _, entry := range entries {
|
||||||
|
// 匹配文件名
|
||||||
|
matched, err := filepath.Match(pattern, entry.Name())
|
||||||
|
if err != nil || !matched {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取文件详细信息
|
||||||
|
info, err := entry.Info()
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// 确定文件类型
|
||||||
|
fileType := "file"
|
||||||
|
if info.IsDir() {
|
||||||
|
fileType = "dir"
|
||||||
|
} else if info.Mode()&os.ModeSymlink != 0 {
|
||||||
|
fileType = "symlink"
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取系统特定的文件信息
|
||||||
|
linkCount, owner, group := getFileInfo(info)
|
||||||
|
|
||||||
|
// 组装文件信息
|
||||||
|
rows = append(rows, FileListRow{
|
||||||
|
FileMode: info.Mode().String(),
|
||||||
|
FileType: fileType,
|
||||||
|
LinkCount: linkCount,
|
||||||
|
Owner: owner,
|
||||||
|
Group: group,
|
||||||
|
Size: info.Size(),
|
||||||
|
ModifiedTime: info.ModTime().UnixMilli(),
|
||||||
|
FileName: entry.Name(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return rows, nil
|
||||||
|
}
|
||||||
36
src/framework/utils/file/files_unix.go
Normal file
36
src/framework/utils/file/files_unix.go
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
//go:build !windows
|
||||||
|
// +build !windows
|
||||||
|
|
||||||
|
package file
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/user"
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
// getFileInfo 获取系统特定的文件信息s
|
||||||
|
func getFileInfo(info os.FileInfo) (linkCount int64, owner, group string) {
|
||||||
|
// Unix-like 系统 (Linux, macOS)
|
||||||
|
if stat, ok := info.Sys().(*syscall.Stat_t); ok {
|
||||||
|
// 获取用户名
|
||||||
|
ownerName := "root"
|
||||||
|
if stat.Uid != 0 {
|
||||||
|
if u, err := user.LookupId(fmt.Sprint(stat.Uid)); err == nil {
|
||||||
|
ownerName = u.Username
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取组名
|
||||||
|
groupName := "root"
|
||||||
|
if stat.Gid != 0 {
|
||||||
|
if g, err := user.LookupGroupId(fmt.Sprint(stat.Gid)); err == nil {
|
||||||
|
groupName = g.Name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return int64(stat.Nlink), ownerName, groupName
|
||||||
|
}
|
||||||
|
return 1, "", ""
|
||||||
|
}
|
||||||
13
src/framework/utils/file/files_windows.go
Normal file
13
src/framework/utils/file/files_windows.go
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
//go:build windows
|
||||||
|
// +build windows
|
||||||
|
|
||||||
|
package file
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
// getFileInfo 获取系统特定的文件信息
|
||||||
|
func getFileInfo(_ os.FileInfo) (linkCount int64, owner, group string) {
|
||||||
|
return 1, "Administrator", "Administrators"
|
||||||
|
}
|
||||||
@@ -37,6 +37,9 @@ func Setup(router *gin.Engine) {
|
|||||||
fileGroup.POST("/chunk-upload", middleware.PreAuthorize(nil), controller.NewFile.ChunkUpload)
|
fileGroup.POST("/chunk-upload", middleware.PreAuthorize(nil), controller.NewFile.ChunkUpload)
|
||||||
fileGroup.POST("/chunk-merge", middleware.PreAuthorize(nil), controller.NewFile.ChunkMerge)
|
fileGroup.POST("/chunk-merge", middleware.PreAuthorize(nil), controller.NewFile.ChunkMerge)
|
||||||
fileGroup.GET("/download/:filePath", middleware.PreAuthorize(nil), controller.NewFile.Download)
|
fileGroup.GET("/download/:filePath", middleware.PreAuthorize(nil), controller.NewFile.Download)
|
||||||
|
fileGroup.GET("/list", middleware.PreAuthorize(nil), controller.NewFile.List)
|
||||||
|
fileGroup.GET("", middleware.PreAuthorize(nil), controller.NewFile.File)
|
||||||
|
fileGroup.DELETE("", middleware.PreAuthorize(nil), controller.NewFile.Remove)
|
||||||
fileGroup.POST("/transfer-static-file", middleware.PreAuthorize(nil), controller.NewFile.TransferStaticFile)
|
fileGroup.POST("/transfer-static-file", middleware.PreAuthorize(nil), controller.NewFile.TransferStaticFile)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ import (
|
|||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"be.ems/src/framework/config"
|
"be.ems/src/framework/config"
|
||||||
@@ -235,6 +237,176 @@ func (s *FileController) ChunkUpload(c *gin.Context) {
|
|||||||
c.JSON(206, resp.OkData(chunkFilePath))
|
c.JSON(206, resp.OkData(chunkFilePath))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 本地文件列表
|
||||||
|
//
|
||||||
|
// GET /list
|
||||||
|
//
|
||||||
|
// @Tags common/file
|
||||||
|
// @Accept json
|
||||||
|
// @Produce json
|
||||||
|
// @Param path query string true "file path" default(/var/log)
|
||||||
|
// @Param pageNum query number true "pageNum" default(1)
|
||||||
|
// @Param pageSize query number true "pageSize" default(10)
|
||||||
|
// @Param search query string false "search prefix" default(upf)
|
||||||
|
// @Success 200 {object} object "Response Results"
|
||||||
|
// @Security TokenAuth
|
||||||
|
// @Summary Local file list
|
||||||
|
// @Description Local file list
|
||||||
|
// @Router /file/list [get]
|
||||||
|
func (s *FileController) List(c *gin.Context) {
|
||||||
|
var querys struct {
|
||||||
|
Path string `form:"path" binding:"required"`
|
||||||
|
PageNum int64 `form:"pageNum" binding:"required"`
|
||||||
|
PageSize int64 `form:"pageSize" binding:"required"`
|
||||||
|
Search string `form:"search"`
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindQuery(&querys); err != nil {
|
||||||
|
errMsgs := fmt.Sprintf("bind err: %s", resp.FormatBindError(err))
|
||||||
|
c.JSON(422, resp.CodeMsg(40422, errMsgs))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取文件列表
|
||||||
|
localFilePath := querys.Path
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
localFilePath = fmt.Sprintf("C:%s", localFilePath)
|
||||||
|
}
|
||||||
|
rows, err := file.FileList(localFilePath, querys.Search)
|
||||||
|
if err != nil {
|
||||||
|
c.JSON(200, resp.OkData(map[string]any{
|
||||||
|
"path": querys.Path,
|
||||||
|
"total": len(rows),
|
||||||
|
"rows": []file.FileListRow{},
|
||||||
|
}))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 对数组进行切片分页
|
||||||
|
lenNum := int64(len(rows))
|
||||||
|
start := (querys.PageNum - 1) * querys.PageSize
|
||||||
|
end := start + querys.PageSize
|
||||||
|
var splitRows []file.FileListRow
|
||||||
|
if start >= lenNum {
|
||||||
|
splitRows = []file.FileListRow{}
|
||||||
|
} else if end >= lenNum {
|
||||||
|
splitRows = rows[start:]
|
||||||
|
} else {
|
||||||
|
splitRows = rows[start:end]
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(200, resp.OkData(map[string]any{
|
||||||
|
"path": querys.Path,
|
||||||
|
"total": lenNum,
|
||||||
|
"rows": splitRows,
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
// 本地文件获取下载
|
||||||
|
//
|
||||||
|
// DELETE /
|
||||||
|
//
|
||||||
|
// @Tags common/file
|
||||||
|
// @Accept json
|
||||||
|
// @Produce json
|
||||||
|
// @Param path query string true "file path" default(/var/log)
|
||||||
|
// @Param fileName query string true "file name" default(omc.log)
|
||||||
|
// @Success 200 {object} object "Response Results"
|
||||||
|
// @Security TokenAuth
|
||||||
|
// @Summary Local files for download
|
||||||
|
// @Description Local files for download
|
||||||
|
// @Router /file [get]
|
||||||
|
func (s *FileController) File(c *gin.Context) {
|
||||||
|
var querys struct {
|
||||||
|
Path string `form:"path" binding:"required"`
|
||||||
|
Filename string `form:"fileName" binding:"required"`
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindQuery(&querys); err != nil {
|
||||||
|
errMsgs := fmt.Sprintf("bind err: %s", resp.FormatBindError(err))
|
||||||
|
c.JSON(422, resp.CodeMsg(40422, errMsgs))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查路径是否在允许的目录范围内
|
||||||
|
allowedPaths := []string{"/var/log", "/tmp", "/usr/local/omc/backup"}
|
||||||
|
allowed := false
|
||||||
|
for _, p := range allowedPaths {
|
||||||
|
if strings.HasPrefix(querys.Path, p) {
|
||||||
|
allowed = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !allowed {
|
||||||
|
c.JSON(200, resp.ErrMsg("operation path is not within the allowed range"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取文件路径并下载
|
||||||
|
localFilePath := filepath.Join(querys.Path, querys.Filename)
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
localFilePath = fmt.Sprintf("C:%s", localFilePath)
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(localFilePath); os.IsNotExist(err) {
|
||||||
|
c.JSON(200, resp.ErrMsg("file does not exist"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
c.FileAttachment(localFilePath, querys.Filename)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 本地文件删除
|
||||||
|
//
|
||||||
|
// DELETE /
|
||||||
|
//
|
||||||
|
// @Tags common/file
|
||||||
|
// @Accept json
|
||||||
|
// @Produce json
|
||||||
|
// @Param path query string true "file path" default(/var/log)
|
||||||
|
// @Param fileName query string true "file name" default(omc.log)
|
||||||
|
// @Success 200 {object} object "Response Results"
|
||||||
|
// @Security TokenAuth
|
||||||
|
// @Summary Local file deletion
|
||||||
|
// @Description Local file deletion
|
||||||
|
// @Router /file [delete]
|
||||||
|
func (s *FileController) Remove(c *gin.Context) {
|
||||||
|
var querys struct {
|
||||||
|
Path string `form:"path" binding:"required"`
|
||||||
|
Filename string `form:"fileName" binding:"required"`
|
||||||
|
}
|
||||||
|
if err := c.ShouldBindQuery(&querys); err != nil {
|
||||||
|
errMsgs := fmt.Sprintf("bind err: %s", resp.FormatBindError(err))
|
||||||
|
c.JSON(422, resp.CodeMsg(40422, errMsgs))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 检查路径是否在允许的目录范围内
|
||||||
|
allowedPaths := []string{"/tmp", "/usr/local/omc/backup"}
|
||||||
|
allowed := false
|
||||||
|
for _, p := range allowedPaths {
|
||||||
|
if strings.HasPrefix(querys.Path, p) {
|
||||||
|
allowed = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !allowed {
|
||||||
|
c.JSON(200, resp.ErrMsg("operation path is not within the allowed range"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取文件路径并删除
|
||||||
|
localFilePath := filepath.Join(querys.Path, querys.Filename)
|
||||||
|
if runtime.GOOS == "windows" {
|
||||||
|
localFilePath = fmt.Sprintf("C:%s", localFilePath)
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(localFilePath); os.IsNotExist(err) {
|
||||||
|
c.JSON(200, resp.ErrMsg("file does not exist"))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if err := os.Remove(localFilePath); err != nil {
|
||||||
|
c.JSON(200, resp.ErrMsg(err.Error()))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
c.JSON(200, resp.Ok(nil))
|
||||||
|
}
|
||||||
|
|
||||||
// 转存指定对应文件到静态目录
|
// 转存指定对应文件到静态目录
|
||||||
//
|
//
|
||||||
// POST /transfer-static-file
|
// POST /transfer-static-file
|
||||||
|
|||||||
Reference in New Issue
Block a user