mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-03 07:12:27 +00:00
parent
5e9c7bc022
commit
7e4e2f3720
@ -1545,7 +1545,7 @@ func (s *service) postFolderVersionsRestore(w http.ResponseWriter, r *http.Reque
|
|||||||
http.Error(w, err.Error(), 500)
|
http.Error(w, err.Error(), 500)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
sendJSON(w, ferr)
|
sendJSON(w, errorStringMap(ferr))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *service) getFolderErrors(w http.ResponseWriter, r *http.Request) {
|
func (s *service) getFolderErrors(w http.ResponseWriter, r *http.Request) {
|
||||||
@ -1839,6 +1839,14 @@ func shouldRegenerateCertificate(cert tls.Certificate) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func errorStringMap(errs map[string]error) map[string]*string {
|
||||||
|
out := make(map[string]*string, len(errs))
|
||||||
|
for s, e := range errs {
|
||||||
|
out[s] = errorString(e)
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
func errorString(err error) *string {
|
func errorString(err error) *string {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
msg := err.Error()
|
msg := err.Error()
|
||||||
|
@ -96,7 +96,7 @@ func (m *mockedModel) GetFolderVersions(folder string) (map[string][]versioner.F
|
|||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *mockedModel) RestoreFolderVersions(folder string, versions map[string]time.Time) (map[string]string, error) {
|
func (m *mockedModel) RestoreFolderVersions(folder string, versions map[string]time.Time) (map[string]error, error) {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,7 +88,7 @@ type Model interface {
|
|||||||
SetIgnores(folder string, content []string) error
|
SetIgnores(folder string, content []string) error
|
||||||
|
|
||||||
GetFolderVersions(folder string) (map[string][]versioner.FileVersion, error)
|
GetFolderVersions(folder string) (map[string][]versioner.FileVersion, error)
|
||||||
RestoreFolderVersions(folder string, versions map[string]time.Time) (map[string]string, error)
|
RestoreFolderVersions(folder string, versions map[string]time.Time) (map[string]error, error)
|
||||||
|
|
||||||
DBSnapshot(folder string) (*db.Snapshot, error)
|
DBSnapshot(folder string) (*db.Snapshot, error)
|
||||||
NeedFolderFiles(folder string, page, perpage int) ([]db.FileInfoTruncated, []db.FileInfoTruncated, []db.FileInfoTruncated, error)
|
NeedFolderFiles(folder string, page, perpage int) ([]db.FileInfoTruncated, []db.FileInfoTruncated, []db.FileInfoTruncated, error)
|
||||||
@ -2624,7 +2624,7 @@ func (m *model) GetFolderVersions(folder string) (map[string][]versioner.FileVer
|
|||||||
return ver.GetVersions()
|
return ver.GetVersions()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *model) RestoreFolderVersions(folder string, versions map[string]time.Time) (map[string]string, error) {
|
func (m *model) RestoreFolderVersions(folder string, versions map[string]time.Time) (map[string]error, error) {
|
||||||
m.fmut.RLock()
|
m.fmut.RLock()
|
||||||
err := m.checkFolderRunningLocked(folder)
|
err := m.checkFolderRunningLocked(folder)
|
||||||
fcfg := m.folderCfgs[folder]
|
fcfg := m.folderCfgs[folder]
|
||||||
@ -2637,11 +2637,11 @@ func (m *model) RestoreFolderVersions(folder string, versions map[string]time.Ti
|
|||||||
return nil, errNoVersioner
|
return nil, errNoVersioner
|
||||||
}
|
}
|
||||||
|
|
||||||
restoreErrors := make(map[string]string)
|
restoreErrors := make(map[string]error)
|
||||||
|
|
||||||
for file, version := range versions {
|
for file, version := range versions {
|
||||||
if err := ver.Restore(file, version); err != nil {
|
if err := ver.Restore(file, version); err != nil {
|
||||||
restoreErrors[file] = err.Error()
|
restoreErrors[file] = err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -26,6 +26,7 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
"github.com/syncthing/syncthing/lib/config"
|
"github.com/syncthing/syncthing/lib/config"
|
||||||
"github.com/syncthing/syncthing/lib/db"
|
"github.com/syncthing/syncthing/lib/db"
|
||||||
"github.com/syncthing/syncthing/lib/db/backend"
|
"github.com/syncthing/syncthing/lib/db/backend"
|
||||||
@ -2859,7 +2860,7 @@ func TestVersionRestore(t *testing.T) {
|
|||||||
ferr, err := m.RestoreFolderVersions("default", restore)
|
ferr, err := m.RestoreFolderVersions("default", restore)
|
||||||
must(t, err)
|
must(t, err)
|
||||||
|
|
||||||
if err, ok := ferr["something"]; len(ferr) > 1 || !ok || err != "cannot restore on top of a directory" {
|
if err, ok := ferr["something"]; len(ferr) > 1 || !ok || !errors.Is(err, versioner.ErrDirectory) {
|
||||||
t.Fatalf("incorrect error or count: %d %s", len(ferr), ferr)
|
t.Fatalf("incorrect error or count: %d %s", len(ferr), ferr)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,6 +9,7 @@ package versioner
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"runtime"
|
"runtime"
|
||||||
@ -64,12 +65,12 @@ func (v external) Archive(filePath string) error {
|
|||||||
l.Debugln("archiving", filePath)
|
l.Debugln("archiving", filePath)
|
||||||
|
|
||||||
if v.command == "" {
|
if v.command == "" {
|
||||||
return errors.New("Versioner: command is empty, please enter a valid command")
|
return errors.New("command is empty, please enter a valid command")
|
||||||
}
|
}
|
||||||
|
|
||||||
words, err := shellquote.Split(v.command)
|
words, err := shellquote.Split(v.command)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errors.New("Versioner: command is invalid: " + err.Error())
|
return fmt.Errorf("command is invalid: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
context := map[string]string{
|
context := map[string]string{
|
||||||
@ -99,6 +100,9 @@ func (v external) Archive(filePath string) error {
|
|||||||
combinedOutput, err := cmd.CombinedOutput()
|
combinedOutput, err := cmd.CombinedOutput()
|
||||||
l.Debugln("external command output:", string(combinedOutput))
|
l.Debugln("external command output:", string(combinedOutput))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
if eerr, ok := err.(*exec.ExitError); ok && len(eerr.Stderr) > 0 {
|
||||||
|
return fmt.Errorf("%v: %v", err, string(eerr.Stderr))
|
||||||
|
}
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,7 +110,7 @@ func (v external) Archive(filePath string) error {
|
|||||||
if _, err = v.filesystem.Lstat(filePath); fs.IsNotExist(err) {
|
if _, err = v.filesystem.Lstat(filePath); fs.IsNotExist(err) {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return errors.New("Versioner: file was not removed by external script")
|
return errors.New("file was not removed by external script")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v external) GetVersions() (map[string][]FileVersion, error) {
|
func (v external) GetVersions() (map[string][]FileVersion, error) {
|
||||||
|
@ -23,7 +23,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
errDirectory = errors.New("cannot restore on top of a directory")
|
ErrDirectory = errors.New("cannot restore on top of a directory")
|
||||||
errNotFound = errors.New("version not found")
|
errNotFound = errors.New("version not found")
|
||||||
errFileAlreadyExists = errors.New("file already exists")
|
errFileAlreadyExists = errors.New("file already exists")
|
||||||
)
|
)
|
||||||
@ -196,7 +196,7 @@ func restoreFile(method fs.CopyRangeMethod, src, dst fs.Filesystem, filePath str
|
|||||||
if info, err := dst.Lstat(filePath); err == nil {
|
if info, err := dst.Lstat(filePath); err == nil {
|
||||||
switch {
|
switch {
|
||||||
case info.IsDir():
|
case info.IsDir():
|
||||||
return errDirectory
|
return ErrDirectory
|
||||||
case info.IsSymlink():
|
case info.IsSymlink():
|
||||||
// Remove existing symlinks (as we don't want to archive them)
|
// Remove existing symlinks (as we don't want to archive them)
|
||||||
if err := dst.Remove(filePath); err != nil {
|
if err := dst.Remove(filePath); err != nil {
|
||||||
|
@ -47,5 +47,37 @@ func New(cfg config.FolderConfiguration) (Versioner, error) {
|
|||||||
return nil, fmt.Errorf("requested versioning type %q does not exist", cfg.Type)
|
return nil, fmt.Errorf("requested versioning type %q does not exist", cfg.Type)
|
||||||
}
|
}
|
||||||
|
|
||||||
return fac(cfg), nil
|
return &versionerWithErrorContext{
|
||||||
|
Versioner: fac(cfg),
|
||||||
|
vtype: cfg.Versioning.Type,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type versionerWithErrorContext struct {
|
||||||
|
Versioner
|
||||||
|
vtype string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *versionerWithErrorContext) wrapError(err error, op string) error {
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("%s versioner: %v: %w", v.vtype, op, err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *versionerWithErrorContext) Archive(filePath string) error {
|
||||||
|
return v.wrapError(v.Versioner.Archive(filePath), "archive")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *versionerWithErrorContext) GetVersions() (map[string][]FileVersion, error) {
|
||||||
|
versions, err := v.Versioner.GetVersions()
|
||||||
|
return versions, v.wrapError(err, "get versions")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *versionerWithErrorContext) Restore(filePath string, versionTime time.Time) error {
|
||||||
|
return v.wrapError(v.Versioner.Restore(filePath, versionTime), "restore")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *versionerWithErrorContext) Clean(ctx context.Context) error {
|
||||||
|
return v.wrapError(v.Versioner.Clean(ctx), "clean")
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user