mirror of
https://github.com/nxshock/backuper.git
synced 2024-11-28 00:21:02 +05:00
272 lines
5.8 KiB
Go
272 lines
5.8 KiB
Go
package main
|
|
|
|
import (
|
|
"archive/tar"
|
|
"encoding/csv"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"sort"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/klauspost/compress/zstd"
|
|
)
|
|
|
|
type FileInfo struct {
|
|
ArchiveFileName string
|
|
ModificationTime time.Time
|
|
|
|
filePath string
|
|
fileSize int64
|
|
}
|
|
|
|
type Index map[string]FileHistory
|
|
|
|
func (index Index) AddFile(fileName string, archiveFileName string, modTime time.Time) {
|
|
fileInfo := FileInfo{ArchiveFileName: archiveFileName, ModificationTime: modTime}
|
|
|
|
if eFileInfo, exists := index[fileName]; exists {
|
|
index[fileName] = append(eFileInfo, fileInfo)
|
|
return
|
|
}
|
|
|
|
index[fileName] = FileHistory{fileInfo}
|
|
}
|
|
|
|
func (index Index) ViewFileVersions(w io.Writer) error {
|
|
for filePath, fileHistory := range index {
|
|
_, err := fmt.Fprintf(w, "%s\n", filePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, v := range fileHistory {
|
|
_, err := fmt.Fprintf(w, "\t%s %s\n", v.ModificationTime.Format(defaultTimeFormat), v.ArchiveFileName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (index Index) Save(fileName string) error {
|
|
f, err := os.Create(fileName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
enc, err := zstd.NewWriter(f, zstd.WithEncoderLevel(zstd.SpeedBestCompression))
|
|
if err != nil {
|
|
f.Close()
|
|
os.Remove(fileName)
|
|
return err
|
|
}
|
|
|
|
files := make([]string, 0, len(index))
|
|
for fileName := range index {
|
|
files = append(files, fileName)
|
|
}
|
|
|
|
// Sort file list for better compression
|
|
sort.Strings(files)
|
|
|
|
csvWriter := csv.NewWriter(enc)
|
|
csvWriter.Comma = ';'
|
|
|
|
for _, fileName := range files {
|
|
for _, historyItem := range index[fileName] {
|
|
err := csvWriter.Write([]string{fileName, historyItem.ArchiveFileName, strconv.Itoa(int(historyItem.ModificationTime.Unix()))})
|
|
if err != nil {
|
|
enc.Close()
|
|
f.Close()
|
|
os.Remove(fileName)
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
csvWriter.Flush()
|
|
if err := csvWriter.Error(); err != nil {
|
|
enc.Close()
|
|
f.Close()
|
|
os.Remove(fileName)
|
|
return err
|
|
}
|
|
|
|
err = enc.Close()
|
|
if err != nil {
|
|
f.Close()
|
|
os.Remove(fileName)
|
|
return err
|
|
}
|
|
|
|
err = f.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
func (b *Config) index(fullIndex bool) (Index, error) {
|
|
index, err := b.indexFromFile()
|
|
if err == nil {
|
|
b.logf(Debug, "Index file contains %d of files.", len(index))
|
|
return index, nil
|
|
}
|
|
b.logf(Error, "index file read error: %v", err)
|
|
|
|
return b.indexFromDisk(fullIndex)
|
|
}
|
|
|
|
func (b *Config) indexFromFile() (Index, error) {
|
|
index := make(Index)
|
|
|
|
indexFileName := filepath.Join(filepath.Dir(b.filePath), indexFileName)
|
|
|
|
f, err := os.Open(indexFileName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer f.Close()
|
|
|
|
dec, err := zstd.NewReader(f)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer dec.Close()
|
|
|
|
csvReader := csv.NewReader(dec)
|
|
csvReader.Comma = ';'
|
|
csvReader.FieldsPerRecord = 3
|
|
for {
|
|
data, err := csvReader.Read()
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
unixTime, err := strconv.Atoi(data[2])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
index.AddFile(data[0], data[1], time.Unix(int64(unixTime), 0).Local())
|
|
}
|
|
|
|
return index, nil
|
|
}
|
|
|
|
func (b *Config) indexFromDisk(fullIndex bool) (Index, error) {
|
|
b.logf(Info, "Rebuilding index from %s...", filepath.Dir(b.filePath))
|
|
allFileMask := filepath.Join(filepath.Dir(b.filePath), b.FileName+"*"+defaultExt)
|
|
onlyFullBackupFileMask := filepath.Join(filepath.Dir(b.filePath), b.FileName+"*f"+defaultExt)
|
|
|
|
// Get last full backup name
|
|
lastFullBackupFileName := ""
|
|
err := filepath.WalkDir(filepath.Dir(b.filePath), func(path string, info os.DirEntry, err error) error {
|
|
matched, err := filepath.Match(onlyFullBackupFileMask, path)
|
|
if err != nil {
|
|
return fmt.Errorf("filepath.WalkDir: %v", err)
|
|
}
|
|
if !matched {
|
|
return nil
|
|
}
|
|
|
|
lastFullBackupFileName = path
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return nil, fmt.Errorf("filepath.WalkDir: %v", err)
|
|
}
|
|
|
|
if !fullIndex {
|
|
b.logf(Debug, "Diff will be calculated from %s.", filepath.Base(lastFullBackupFileName))
|
|
}
|
|
|
|
var files []string
|
|
err = filepath.WalkDir(filepath.Dir(b.filePath), func(path string, info os.DirEntry, err error) error {
|
|
matched, err := filepath.Match(allFileMask, path)
|
|
if err != nil {
|
|
return fmt.Errorf("filepath.Match: %v", err)
|
|
}
|
|
|
|
if matched && (fullIndex || path >= lastFullBackupFileName) {
|
|
files = append(files, path)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return nil, fmt.Errorf("filepath.Walk: %v", err)
|
|
}
|
|
|
|
index := make(Index)
|
|
|
|
for i, file := range files {
|
|
b.logf(Debug, "[%3d%%] Reading file %s...", (100 * i / len(files)), filepath.Base(file))
|
|
f, err := os.Open(file)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("os.Open: %v", err)
|
|
}
|
|
defer f.Close()
|
|
|
|
decoder, err := zstd.NewReader(f)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("zstd.NewReader: %v", err)
|
|
}
|
|
|
|
tarReader := tar.NewReader(decoder)
|
|
|
|
for {
|
|
tarHeader, err := tarReader.Next()
|
|
if err != nil {
|
|
if err == io.EOF {
|
|
break
|
|
} else {
|
|
return nil, fmt.Errorf("ошибка при чтении списка файлов из архива %s: %v", file, err)
|
|
}
|
|
}
|
|
|
|
index[tarHeader.Name] = append(index[tarHeader.Name], FileInfo{
|
|
filePath: tarHeader.Name,
|
|
ModificationTime: tarHeader.FileInfo().ModTime(),
|
|
fileSize: tarHeader.FileInfo().Size(),
|
|
ArchiveFileName: filepath.Base(file)})
|
|
}
|
|
decoder.Close()
|
|
}
|
|
|
|
return index, nil
|
|
}
|
|
|
|
func (index Index) GetFilesLocation(mask string, t time.Time) ([]FileInfo, error) {
|
|
var files2 []FileInfo
|
|
|
|
for fileName := range index {
|
|
if isFilePathMatchPatterns([]string{mask}, fileName) {
|
|
files := index[fileName]
|
|
|
|
file := files[0]
|
|
for i := 1; i < len(files); i++ {
|
|
if files[i].ModificationTime.Before(t) && files[i].ModificationTime.Sub(t) > file.ModificationTime.Sub(t) { // Больше, т.к. отрицательные числа
|
|
file = files[i]
|
|
}
|
|
}
|
|
|
|
file.filePath = fileName
|
|
|
|
files2 = append(files2, file)
|
|
}
|
|
}
|
|
|
|
return files2, nil
|
|
}
|