mirror of
https://gitee.com/XM-GO/PandaX.git
synced 2026-04-23 02:48:34 +08:00
107
kit/file/file.go
107
kit/file/file.go
@@ -5,26 +5,113 @@ import (
|
|||||||
"mime/multipart"
|
"mime/multipart"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
|
"pandax/kit/biz"
|
||||||
|
"strconv"
|
||||||
|
"sync"
|
||||||
)
|
)
|
||||||
|
|
||||||
// DownloadFile 会将url下载到本地文件,它会在下载时写入,而不是将整个文件加载到内存中。
|
const (
|
||||||
func DownloadFile(url, filepath string) error {
|
MaxConcurrency = 16 // 最大并发数
|
||||||
// Get the data
|
)
|
||||||
resp, err := http.Get(url)
|
|
||||||
|
type DownloadTask struct {
|
||||||
|
URL string
|
||||||
|
FilePath string
|
||||||
|
}
|
||||||
|
|
||||||
|
func DownloadFileWithConcurrency(url, filepath string) error {
|
||||||
|
resp, err := http.Head(url)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
|
|
||||||
// Create the file
|
fileSize, err := strconv.Atoi(resp.Header.Get("Content-Length"))
|
||||||
out, err := os.Create(filepath)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
defer out.Close()
|
|
||||||
// Write the body to file
|
file, err := os.OpenFile(filepath, os.O_RDWR|os.O_CREATE, 0666)
|
||||||
_, err = io.Copy(out, resp.Body)
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
// 检查本地文件大小
|
||||||
|
localFileSize, err := file.Seek(0, io.SeekEnd)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// 计算剩余未下载的文件大小
|
||||||
|
remainingSize := fileSize - int(localFileSize)
|
||||||
|
|
||||||
|
// 计算每个片段的大小
|
||||||
|
chunkSize := remainingSize / MaxConcurrency
|
||||||
|
|
||||||
|
// 创建等待组,用于等待所有goroutine完成
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
wg.Add(MaxConcurrency)
|
||||||
|
|
||||||
|
// 创建并发下载任务
|
||||||
|
for i := 0; i < MaxConcurrency; i++ {
|
||||||
|
start := localFileSize + int64(i*chunkSize)
|
||||||
|
end := start + int64(chunkSize) - 1
|
||||||
|
|
||||||
|
// 最后一个片段的结束位置可能超过文件大小,需要修正
|
||||||
|
if i == MaxConcurrency-1 {
|
||||||
|
end = int64(fileSize) - 1
|
||||||
|
}
|
||||||
|
|
||||||
|
go func(index int, start, end int64) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
err := downloadChunk(url, filepath, start, end)
|
||||||
|
if err != nil {
|
||||||
|
biz.NewBizErr("文件下载失败")
|
||||||
|
// 处理下载错误
|
||||||
|
}
|
||||||
|
}(i, start, end)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 等待所有goroutine完成
|
||||||
|
wg.Wait()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func downloadChunk(url, filepath string, start, end int64) error {
|
||||||
|
req, err := http.NewRequest("GET", url, nil)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// 设置Range头部
|
||||||
|
req.Header.Set("Range", "bytes="+strconv.FormatInt(start, 10)+"-"+strconv.FormatInt(end, 10))
|
||||||
|
|
||||||
|
resp, err := http.DefaultClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
file, err := os.OpenFile(filepath, os.O_RDWR, 0666)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
_, err = file.Seek(start, io.SeekStart)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = io.CopyN(file, resp.Body, end-start+1)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func SaveUploadedFile(file *multipart.FileHeader, dst string) error {
|
func SaveUploadedFile(file *multipart.FileHeader, dst string) error {
|
||||||
|
|||||||
Reference in New Issue
Block a user