Skip to content

Commit

Permalink
Fix moveFiles for zip files (#3608)
Browse files Browse the repository at this point in the history
  • Loading branch information
DingDongSoLong4 authored Mar 28, 2023
1 parent 046fd1c commit a8f9310
Show file tree
Hide file tree
Showing 5 changed files with 109 additions and 12 deletions.
15 changes: 10 additions & 5 deletions internal/api/resolver_mutation_file.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,9 @@ import (

func (r *mutationResolver) MoveFiles(ctx context.Context, input MoveFilesInput) (bool, error) {
if err := r.withTxn(ctx, func(ctx context.Context) error {
qb := r.repository.File
mover := file.NewMover(qb)
fileStore := r.repository.File
folderStore := r.repository.Folder
mover := file.NewMover(fileStore, folderStore)
mover.RegisterHooks(ctx, r.txnManager)

var (
Expand All @@ -36,14 +37,18 @@ func (r *mutationResolver) MoveFiles(ctx context.Context, input MoveFilesInput)
return fmt.Errorf("invalid folder id %s: %w", *input.DestinationFolderID, err)
}

folder, err = r.repository.Folder.Find(ctx, file.FolderID(folderID))
folder, err = folderStore.Find(ctx, file.FolderID(folderID))
if err != nil {
return fmt.Errorf("finding destination folder: %w", err)
}

if folder == nil {
return fmt.Errorf("folder with id %d not found", input.DestinationFolderID)
}

if folder.ZipFileID != nil {
return fmt.Errorf("cannot move to %s, is in a zip file", folder.Path)
}
case input.DestinationFolder != nil:
folderPath := *input.DestinationFolder

Expand All @@ -54,7 +59,7 @@ func (r *mutationResolver) MoveFiles(ctx context.Context, input MoveFilesInput)

// get or create folder hierarchy
var err error
folder, err = file.GetOrCreateFolderHierarchy(ctx, r.repository.Folder, folderPath)
folder, err = file.GetOrCreateFolderHierarchy(ctx, folderStore, folderPath)
if err != nil {
return fmt.Errorf("getting or creating folder hierarchy: %w", err)
}
Expand All @@ -78,7 +83,7 @@ func (r *mutationResolver) MoveFiles(ctx context.Context, input MoveFilesInput)

for _, fileIDInt := range fileIDs {
fileID := file.ID(fileIDInt)
f, err := qb.Find(ctx, fileID)
f, err := fileStore.Find(ctx, fileID)
if err != nil {
return fmt.Errorf("finding file %d: %w", fileID, err)
}
Expand Down
2 changes: 0 additions & 2 deletions internal/manager/repository.go
Original file line number Diff line number Diff line change
Expand Up @@ -35,15 +35,13 @@ type SceneReaderWriter interface {

type FileReaderWriter interface {
file.Store
file.Finder
Query(ctx context.Context, options models.FileQueryOptions) (*models.FileQueryResult, error)
GetCaptions(ctx context.Context, fileID file.ID) ([]*models.VideoCaption, error)
IsPrimary(ctx context.Context, fileID file.ID) (bool, error)
}

type FolderReaderWriter interface {
file.FolderStore
Find(ctx context.Context, id file.FolderID) (*file.Folder, error)
}

type Repository struct {
Expand Down
5 changes: 5 additions & 0 deletions pkg/file/file.go
Original file line number Diff line number Diff line change
Expand Up @@ -180,6 +180,11 @@ type Destroyer interface {
Destroy(ctx context.Context, id ID) error
}

type GetterUpdater interface {
Getter
Updater
}

type GetterDestroyer interface {
Getter
Destroyer
Expand Down
5 changes: 5 additions & 0 deletions pkg/file/folder.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,13 +32,18 @@ func (f *Folder) Info(fs FS) (fs.FileInfo, error) {
return f.info(fs, f.Path)
}

type FolderFinder interface {
Find(ctx context.Context, id FolderID) (*Folder, error)
}

// FolderPathFinder finds Folders by their path.
type FolderPathFinder interface {
FindByPath(ctx context.Context, path string) (*Folder, error)
}

// FolderGetter provides methods to find Folders.
type FolderGetter interface {
FolderFinder
FolderPathFinder
FindByZipFileID(ctx context.Context, zipFileID ID) ([]*Folder, error)
FindAllInPaths(ctx context.Context, p []string, limit, offset int) ([]*Folder, error)
Expand Down
94 changes: 89 additions & 5 deletions pkg/file/move.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import (
"io/fs"
"os"
"path/filepath"
"strings"
"time"

"github.com/stashapp/stash/pkg/logger"
Expand Down Expand Up @@ -39,15 +40,17 @@ func (r folderCreatorStatRenamerImpl) Mkdir(name string, perm os.FileMode) error

type Mover struct {
Renamer DirMakerStatRenamer
Updater Updater
Files GetterUpdater
Folders FolderStore

moved map[string]string
foldersCreated []string
}

func NewMover(u Updater) *Mover {
func NewMover(fileStore GetterUpdater, folderStore FolderStore) *Mover {
return &Mover{
Updater: u,
Files: fileStore,
Folders: folderStore,
Renamer: &folderCreatorStatRenamerImpl{
renamerRemoverImpl: newRenamerRemoverImpl(),
mkDirFn: os.Mkdir,
Expand All @@ -62,7 +65,7 @@ func (m *Mover) Move(ctx context.Context, f File, folder *Folder, basename strin

// don't allow moving files in zip files
if fBase.ZipFileID != nil {
return fmt.Errorf("cannot move file %s in zip file", f.Base().Path)
return fmt.Errorf("cannot move file %s, is in a zip file", fBase.Path)
}

if basename == "" {
Expand All @@ -84,12 +87,50 @@ func (m *Mover) Move(ctx context.Context, f File, folder *Folder, basename strin
return fmt.Errorf("file %s already exists", newPath)
}

if err := m.transferZipFolderHierarchy(ctx, fBase.ID, oldPath, newPath); err != nil {
return fmt.Errorf("moving folder hierarchy for file %s: %w", fBase.Path, err)
}

// move contained files if file is a zip file
zipFiles, err := m.Files.FindByZipFileID(ctx, fBase.ID)
if err != nil {
return fmt.Errorf("finding contained files in file %s: %w", fBase.Path, err)
}
for _, zf := range zipFiles {
zfBase := zf.Base()
oldZfPath := zfBase.Path
oldZfDir := filepath.Dir(oldZfPath)

// sanity check - ignore files which aren't under oldPath
if !strings.HasPrefix(oldZfPath, oldPath) {
continue
}

relZfDir, err := filepath.Rel(oldPath, oldZfDir)
if err != nil {
return fmt.Errorf("moving contained file %s: %w", zfBase.ID, err)
}
newZfDir := filepath.Join(newPath, relZfDir)

// folder should have been created by moveZipFolderHierarchy
newZfFolder, err := GetOrCreateFolderHierarchy(ctx, m.Folders, newZfDir)
if err != nil {
return fmt.Errorf("getting or creating folder hierarchy: %w", err)
}

// update file parent folder
zfBase.ParentFolderID = newZfFolder.ID
if err := m.Files.Update(ctx, zf); err != nil {
return fmt.Errorf("updating file %s: %w", oldZfPath, err)
}
}

fBase.ParentFolderID = folder.ID
fBase.Basename = basename
fBase.UpdatedAt = time.Now()
// leave ModTime as is. It may or may not be changed by this operation

if err := m.Updater.Update(ctx, f); err != nil {
if err := m.Files.Update(ctx, f); err != nil {
return fmt.Errorf("updating file %s: %w", oldPath, err)
}

Expand Down Expand Up @@ -125,6 +166,49 @@ func (m *Mover) CreateFolderHierarchy(path string) error {
return nil
}

// transferZipFolderHierarchy creates the folder hierarchy for zipFileID under newPath, and removes
// ZipFileID from folders under oldPath.
func (m *Mover) transferZipFolderHierarchy(ctx context.Context, zipFileID ID, oldPath string, newPath string) error {
zipFolders, err := m.Folders.FindByZipFileID(ctx, zipFileID)
if err != nil {
return err
}

for _, oldFolder := range zipFolders {
oldZfPath := oldFolder.Path

// sanity check - ignore folders which aren't under oldPath
if !strings.HasPrefix(oldZfPath, oldPath) {
continue
}

relZfPath, err := filepath.Rel(oldPath, oldZfPath)
if err != nil {
return err
}
newZfPath := filepath.Join(newPath, relZfPath)

newFolder, err := GetOrCreateFolderHierarchy(ctx, m.Folders, newZfPath)
if err != nil {
return err
}

// add ZipFileID to new folder
newFolder.ZipFileID = &zipFileID
if err = m.Folders.Update(ctx, newFolder); err != nil {
return err
}

// remove ZipFileID from old folder
oldFolder.ZipFileID = nil
if err = m.Folders.Update(ctx, oldFolder); err != nil {
return err
}
}

return nil
}

func (m *Mover) moveFile(oldPath, newPath string) error {
if err := m.Renamer.Rename(oldPath, newPath); err != nil {
return fmt.Errorf("renaming file %s to %s: %w", oldPath, newPath, err)
Expand Down

0 comments on commit a8f9310

Please sign in to comment.