Cloudreve/pkg/filesystem/archive.go

129 lines
3.1 KiB
Go
Raw Normal View History

2019-12-11 18:52:41 +08:00
package filesystem
import (
2019-12-13 09:46:53 +08:00
"archive/zip"
"context"
"fmt"
2019-12-13 09:46:53 +08:00
model "github.com/HFO4/cloudreve/models"
"github.com/HFO4/cloudreve/pkg/util"
"io"
"path/filepath"
"time"
)
2019-12-13 09:46:53 +08:00
/* ===============
压缩/解压缩
2019-12-13 09:46:53 +08:00
===============
*/
// Compress 创建给定目录和文件的压缩文件
func (fs *FileSystem) Compress(ctx context.Context, folderIDs, fileIDs []uint) (string, error) {
2019-12-13 09:46:53 +08:00
// 查找待压缩目录
folders, err := model.GetFoldersByIDs(folderIDs, fs.User.ID)
if err != nil && len(folders) != 0 {
return "", ErrDBListObjects
2019-12-13 09:46:53 +08:00
}
// 查找待压缩文件
files, err := model.GetFilesByIDs(fileIDs, fs.User.ID)
if err != nil && len(files) != 0 {
return "", ErrDBListObjects
2019-12-13 09:46:53 +08:00
}
// 将顶级待处理对象的路径设为根路径
for i := 0; i < len(folders); i++ {
folders[i].Position = ""
}
for i := 0; i < len(files); i++ {
files[i].Position = ""
}
// 创建临时压缩文件
zipFilePath := filepath.Join(
model.GetSettingByName("temp_path"),
fmt.Sprintf("archive_%d.zip", time.Now().UnixNano()),
)
zipFile, err := util.CreatNestedFile(zipFilePath)
2019-12-13 09:46:53 +08:00
if err != nil {
util.Log().Warning("%s", err)
return "", err
2019-12-13 09:46:53 +08:00
}
defer zipFile.Close()
2019-12-13 09:46:53 +08:00
// 创建压缩文件Writer
zipWriter := zip.NewWriter(zipFile)
2019-12-13 09:46:53 +08:00
defer zipWriter.Close()
ctx, _ = context.WithCancel(context.Background())
2020-01-02 14:51:17 +08:00
// ctx = context.WithValue(ctx, fsctx.UserCtx, *fs.User)
2019-12-13 09:46:53 +08:00
// 压缩各个目录及文件
for i := 0; i < len(folders); i++ {
fs.doCompress(ctx, nil, &folders[i], zipWriter, true)
2019-12-13 09:46:53 +08:00
}
for i := 0; i < len(files); i++ {
fs.doCompress(ctx, &files[i], nil, zipWriter, true)
2019-12-13 09:46:53 +08:00
}
return zipFilePath, nil
2019-12-13 09:46:53 +08:00
}
func (fs *FileSystem) doCompress(ctx context.Context, file *model.File, folder *model.Folder, zipWriter *zip.Writer, isArchive bool) {
2019-12-13 09:46:53 +08:00
// 如果对象是文件
if file != nil {
// 切换上传策略
fs.Policy = file.GetPolicy()
err := fs.dispatchHandler()
if err != nil {
util.Log().Warning("无法压缩文件%s%s", file.Name, err)
return
}
// 获取文件内容
2019-12-13 09:46:53 +08:00
fileToZip, err := fs.Handler.Get(ctx, file.SourceName)
if err != nil {
util.Log().Debug("Open%s%s", file.Name, err)
return
}
if closer, ok := fileToZip.(io.Closer); ok {
defer closer.Close()
}
// 创建压缩文件头
2019-12-13 09:46:53 +08:00
header := &zip.FileHeader{
Name: filepath.Join(file.Position, file.Name),
Modified: file.UpdatedAt,
UncompressedSize64: file.Size,
}
// 指定是压缩还是归档
if isArchive {
header.Method = zip.Store
} else {
header.Method = zip.Deflate
2019-12-13 09:46:53 +08:00
}
writer, err := zipWriter.CreateHeader(header)
if err != nil {
return
}
_, err = io.Copy(writer, fileToZip)
} else if folder != nil {
// 对象是目录
// 获取子文件
subFiles, err := folder.GetChildFiles()
if err == nil && len(subFiles) > 0 {
for i := 0; i < len(subFiles); i++ {
fs.doCompress(ctx, &subFiles[i], nil, zipWriter, isArchive)
}
}
// 获取子目录,继续递归遍历
subFolders, err := folder.GetChildFolder()
if err == nil && len(subFolders) > 0 {
for i := 0; i < len(subFolders); i++ {
fs.doCompress(ctx, nil, &subFolders[i], zipWriter, isArchive)
}
}
2019-12-13 09:46:53 +08:00
}
}