1
0
mirror of https://github.com/haiwen/seafile-server.git synced 2025-06-26 23:16:51 +00:00
seafile-server/fileserver/merge.go
Jiaqiang Xu 7420b8d738
Go fileserver (#437)
* Initial commit for fileserver written in golang.

[gofileserver] Fix some syntaxt errors.

Add fs backend and objstore test (#352)

* Add fs backend and objstore test

* modify test case and optimize fs backend

* Modify function name and first write temporary files

* Don't need to reopen the temp files

Add comment for objstore (#354)

* Add comment for objstore

* Modify comment

Add commitmgr and test case (#356)

* Add commitmgr and test case

* Redefine the interface

* Modify comment and interface

* Modify parameter and del unused method

* Add comment for FromData and ToData

Add blockmgr and test case (#357)

* Add blockmgr and test case

* Modify comment and interface

Add fsmgr and test case (#358)

* Add fsmgr and test case

* Add save interface and error details

* Modify errors and comments

Add searpc package and test case (#360)

* Add searpc package

* Add searpc test case

* Add return error and add Request struct

* Modify returned error

* Modify comments

add checkPerm (#369)

Add file and block download (#363)

* Add file and block download

* Modify init and use aes algorithm

* Get block by offset and add stat method

* Modify objID's type

* Fix reset pos after add start

* Add http error handing and record log when failed to read block or write block to response

* Modify http return code and value names

* Modify http return code and add log info

* Block read add comment and only repeat once

load ccnetdb and support sqlite (#371)

Add zip download (#372)

* Add zip download

* Modify pack dir and log info

* Modify http return code and use Deflate zip compression methods

add /repo/<repo-id>/permission-check (#375)

add /<repo-id>/commit/HEAD (#377)

add  /repo/<repo-id>/commit/<id> (#379)

add /repo/<repo-id>/block/<id> (#380)

add /repo/<repo-id>/fs-id-list (#383)

add /repo/head-commits-multi (#388)

Add file upload api (#378)

* Add file upload api

* Upload api implements post multi files and create relative path

* Modify handle error and save files directly

* Fix rebase conflict

* index block use channel and optimize mkdir with parents

* Handle jobs and results in a loop

* Mkdir with parents use postMultiFiles and use pointer of SeafDirent

* Del diff_simple size_sched virtual_repo

* Need to check the path with and without slash

* Modify merge trees and add merge test case

* Del postFile and don't close results channel

* Close the file and remove multipart temp file

* Modify merge test case and compare the first name of path

* Use pointer of Entries for SeafDir

* Add test cases for different situations

add /repo/<repo-id>/pack-fs (#389)

add POST /<repo-id>/check-fs and /<repo-id>/check-blocks (#396)

Merge compute repo (#397)

* Add update repo size and merge virtual repo

* Eliminate lint warnings

* Uncomment merge virtual repo and compute repo size

* Need init the dents

* Use interface{} param and modify removeElems

* Move update dir to file.go and modify logs

* Del sync pkg

add PUT /<repo-id>/commit/<commit-id> (#400)

add PUT /<repo-id>/block/<id> (#401)

add POST /<repo-id>/recv-fs (#398)

add PUT /<repo-id>/commit/HEAD (#402)

Add http return code (#403)

Add file update API (#399)

* Add file update API

* Add GetObjIDByPath and fix change size error

* Add traffic statistics for update api

add diffTrees unit test (#391)

add GET /accessible-repos (#406)

add GET /<repo-id>/block-map/<file-id> (#405)

Add test update repo size and merge virtual repo (#409)

* Update dir need update repo size

* Add test update repo size and merge virtual repo

* Add delay for test ajax

* Add delay before get repo size and modify comment

Use go fileserver for unit test (#410)

* Use go fileserver for unit test

* Blocking scheduling update repo size

* Add delay because of sqlite doesn't support concurrency

* Post use multipart form encode

* Del mysql database when test finished

* Fix merge virtual repo failed when use sqlite3

Add upload block API (#412)

fixed error

Add quota-check API (#426)

use diff package

* Use central conf for go fileserver (#428)

* Use central conf for go fileserver

* Fix log error

* use store id and remove share get repo owner (#430)

* Fix permission error (#432)

Co-authored-by: feiniks <36756310+feiniks@users.noreply.github.com>
Co-authored-by: Xiangyue Cai <caixiangyue007@gmail.com>
2021-01-04 11:41:53 +08:00

403 lines
9.8 KiB
Go

package main
import (
"fmt"
"path/filepath"
"sort"
"strings"
"time"
"github.com/haiwen/seafile-server/fileserver/commitmgr"
"github.com/haiwen/seafile-server/fileserver/fsmgr"
)
type mergeOptions struct {
remoteRepoID string
remoteHead string
mergedRoot string
conflict bool
}
func mergeTrees(storeID string, roots []string, opt *mergeOptions) error {
if len(roots) != 3 {
err := fmt.Errorf("invalid argument")
return err
}
var trees []*fsmgr.SeafDir
for i := 0; i < 3; i++ {
dir, err := fsmgr.GetSeafdir(storeID, roots[i])
if err != nil {
err := fmt.Errorf("failed to get dir: %v", err)
return err
}
trees = append(trees, dir)
}
err := mergeTreesRecursive(storeID, trees, "", opt)
if err != nil {
err := fmt.Errorf("failed to merge trees: %v", err)
return err
}
return nil
}
func mergeTreesRecursive(storeID string, trees []*fsmgr.SeafDir, baseDir string, opt *mergeOptions) error {
var ptrs [3][]*fsmgr.SeafDirent
var mergedDents []*fsmgr.SeafDirent
n := 3
for i := 0; i < n; i++ {
if trees[i] != nil {
ptrs[i] = trees[i].Entries
}
}
var done bool
var offset = make([]int, n)
for {
dents := make([]*fsmgr.SeafDirent, n)
var firstName string
done = true
for i := 0; i < n; i++ {
if len(ptrs[i]) > offset[i] {
done = false
dent := ptrs[i][offset[i]]
if firstName == "" {
firstName = dent.Name
} else if dent.Name > firstName {
firstName = dent.Name
}
}
}
if done {
break
}
var nFiles, nDirs int
for i := 0; i < n; i++ {
if len(ptrs[i]) > offset[i] {
dent := ptrs[i][offset[i]]
if firstName == dent.Name {
if fsmgr.IsDir(dent.Mode) {
nDirs++
} else {
nFiles++
}
dents[i] = dent
offset[i]++
}
}
}
if nFiles > 0 {
retDents, err := mergeEntries(storeID, dents, baseDir, opt)
if err != nil {
return err
}
mergedDents = append(mergedDents, retDents...)
}
if nDirs > 0 {
retDents, err := mergeDirectories(storeID, dents, baseDir, opt)
if err != nil {
return err
}
mergedDents = append(mergedDents, retDents...)
}
}
sort.Sort(Dirents(mergedDents))
mergedTree, err := fsmgr.NewSeafdir(1, mergedDents)
if err != nil {
err := fmt.Errorf("failed to new seafdir: %v", err)
return err
}
opt.mergedRoot = mergedTree.DirID
if trees[1] != nil && trees[1].DirID == mergedTree.DirID ||
trees[2] != nil && trees[2].DirID == mergedTree.DirID {
return nil
}
err = fsmgr.SaveSeafdir(storeID, mergedTree)
if err != nil {
err := fmt.Errorf("failed to save merged tree %s/%s", storeID, baseDir)
return err
}
return nil
}
func mergeEntries(storeID string, dents []*fsmgr.SeafDirent, baseDir string, opt *mergeOptions) ([]*fsmgr.SeafDirent, error) {
var mergedDents []*fsmgr.SeafDirent
n := 3
files := make([]*fsmgr.SeafDirent, n)
for i := 0; i < n; i++ {
if dents[i] != nil && !fsmgr.IsDir(dents[i].Mode) {
files[i] = dents[i]
}
}
base := files[0]
head := files[1]
remote := files[2]
if head != nil && remote != nil {
if head.ID == remote.ID {
mergedDents = append(mergedDents, head)
} else if base != nil && base.ID == head.ID {
mergedDents = append(mergedDents, remote)
} else if base != nil && base.ID == remote.ID {
mergedDents = append(mergedDents, head)
} else {
conflictName, _ := mergeConflictFileName(storeID, opt, baseDir, head.Name)
if conflictName == "" {
err := fmt.Errorf("failed to generate conflict file name")
return nil, err
}
dents[2].Name = conflictName
mergedDents = append(mergedDents, remote)
opt.conflict = true
}
} else if base != nil && head == nil && remote != nil {
if base.ID != remote.ID {
if dents[1] != nil {
conflictName, _ := mergeConflictFileName(storeID, opt, baseDir, remote.Name)
if conflictName == "" {
err := fmt.Errorf("failed to generate conflict file name")
return nil, err
}
dents[2].Name = conflictName
mergedDents = append(mergedDents, remote)
opt.conflict = true
} else {
mergedDents = append(mergedDents, remote)
}
}
} else if base != nil && head != nil && remote == nil {
if base.ID != head.ID {
if dents[2] != nil {
conflictName, _ := mergeConflictFileName(storeID, opt, baseDir, dents[2].Name)
if conflictName == "" {
err := fmt.Errorf("failed to generate conflict file name")
return nil, err
}
dents[2].Name = conflictName
mergedDents = append(mergedDents, head)
opt.conflict = true
} else {
mergedDents = append(mergedDents, head)
}
}
} else if base == nil && head == nil && remote != nil {
if dents[1] == nil {
mergedDents = append(mergedDents, remote)
} else if dents[0] != nil && dents[0].ID == dents[1].ID {
mergedDents = append(mergedDents, remote)
} else {
conflictName, _ := mergeConflictFileName(storeID, opt, baseDir, remote.Name)
if conflictName == "" {
err := fmt.Errorf("failed to generate conflict file name")
return nil, err
}
dents[2].Name = conflictName
mergedDents = append(mergedDents, remote)
opt.conflict = true
}
} else if base == nil && head != nil && remote == nil {
if dents[2] == nil {
mergedDents = append(mergedDents, head)
} else if dents[0] != nil && dents[0].ID == dents[2].ID {
mergedDents = append(mergedDents, head)
} else {
conflictName, _ := mergeConflictFileName(storeID, opt, baseDir, dents[2].Name)
if conflictName == "" {
err := fmt.Errorf("failed to generate conflict file name")
return nil, err
}
dents[2].Name = conflictName
mergedDents = append(mergedDents, head)
opt.conflict = true
}
} else if base != nil && head == nil && remote == nil {
}
return mergedDents, nil
}
func mergeDirectories(storeID string, dents []*fsmgr.SeafDirent, baseDir string, opt *mergeOptions) ([]*fsmgr.SeafDirent, error) {
var dirMask int
var mergedDents []*fsmgr.SeafDirent
var dirName string
n := 3
subDirs := make([]*fsmgr.SeafDir, n)
for i := 0; i < n; i++ {
if dents[i] != nil && fsmgr.IsDir(dents[i].Mode) {
dirMask |= 1 << i
}
}
switch dirMask {
case 0:
err := fmt.Errorf("no dirent for merge")
return nil, err
case 1:
return mergedDents, nil
case 2:
mergedDents = append(mergedDents, dents[1])
return mergedDents, nil
case 3:
if dents[0].ID == dents[1].ID {
return mergedDents, nil
}
break
case 4:
mergedDents = append(mergedDents, dents[2])
return mergedDents, nil
case 5:
if dents[0].ID == dents[2].ID {
return mergedDents, nil
}
break
case 6:
case 7:
if dents[1].ID == dents[2].ID {
mergedDents = append(mergedDents, dents[1])
return mergedDents, nil
} else if dents[0] != nil && dents[0].ID == dents[1].ID {
mergedDents = append(mergedDents, dents[2])
return mergedDents, nil
} else if dents[0] != nil && dents[0].ID == dents[2].ID {
mergedDents = append(mergedDents, dents[1])
return mergedDents, nil
}
break
default:
err := fmt.Errorf("wrong dir mask for merge")
return nil, err
}
for i := 0; i < n; i++ {
subDirs[i] = nil
}
for i := 0; i < n; i++ {
if dents[i] != nil && fsmgr.IsDir(dents[i].Mode) {
dir, err := fsmgr.GetSeafdir(storeID, dents[i].ID)
if err != nil {
err := fmt.Errorf("failed to get seafdir %s/%s", storeID, dents[i].ID)
return nil, err
}
subDirs[i] = dir
dirName = dents[i].Name
}
}
newBaseDir := filepath.Join(baseDir, dirName)
newBaseDir = newBaseDir + "/"
err := mergeTreesRecursive(storeID, subDirs, newBaseDir, opt)
if err != nil {
err := fmt.Errorf("failed to merge trees: %v", err)
return nil, err
}
if dirMask == 3 || dirMask == 6 || dirMask == 7 {
dent := dents[1]
dent.ID = opt.mergedRoot
mergedDents = append(mergedDents, dent)
} else if dirMask == 5 {
dent := dents[2]
dent.ID = opt.mergedRoot
mergedDents = append(mergedDents, dent)
}
return mergedDents, nil
}
func mergeConflictFileName(storeID string, opt *mergeOptions, baseDir, fileName string) (string, error) {
var modifier string
var mtime int64
filePath := filepath.Join(baseDir, fileName)
modifier, mtime, err := getFileModifierMtime(opt.remoteRepoID, storeID, opt.remoteHead, filePath)
if err != nil {
commit, err := commitmgr.Load(opt.remoteRepoID, opt.remoteHead)
if err != nil {
err := fmt.Errorf("failed to get head commit")
return "", err
}
modifier = commit.CreatorName
mtime = time.Now().Unix()
}
conflictName := genConflictPath(fileName, modifier, mtime)
return conflictName, nil
}
func genConflictPath(originPath, modifier string, mtime int64) string {
var conflictPath string
now := time.Now()
timeBuf := now.Format("2006-Jan-2-15-04-05")
dot := strings.Index(originPath, ".")
if dot < 0 {
if modifier != "" {
conflictPath = fmt.Sprintf("%s (SFConflict %s %s)",
originPath, modifier, timeBuf)
} else {
conflictPath = fmt.Sprintf("%s (SFConflict %s)",
originPath, timeBuf)
}
} else {
if modifier != "" {
conflictPath = fmt.Sprintf("%s (SFConflict %s %s).%s",
originPath, modifier, timeBuf, originPath[dot+1:])
} else {
conflictPath = fmt.Sprintf("%s (SFConflict %s).%s",
originPath, timeBuf, originPath[dot+1:])
}
}
return conflictPath
}
func getFileModifierMtime(repoID, storeID, head, filePath string) (string, int64, error) {
commit, err := commitmgr.Load(repoID, head)
if err != nil {
err := fmt.Errorf("failed to get head commit")
return "", -1, err
}
parent := filepath.Dir(filePath)
if parent == "." {
parent = ""
}
fileName := filepath.Base(filePath)
dir, err := fsmgr.GetSeafdirByPath(storeID, commit.RootID, parent)
if err != nil {
err := fmt.Errorf("dir %s doesn't exist in repo %s", parent, repoID)
return "", -1, err
}
var dent *fsmgr.SeafDirent
entries := dir.Entries
for _, d := range entries {
if d.Name == fileName {
dent = d
break
}
}
if dent == nil {
err := fmt.Errorf("file %s doesn't exist in repo %s", fileName, repoID)
return "", -1, err
}
return dent.Modifier, dent.Mtime, nil
}