172 lines
4.2 KiB
Go
172 lines
4.2 KiB
Go
package photoprism
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime/debug"
|
|
"strings"
|
|
|
|
"github.com/dustin/go-humanize/english"
|
|
|
|
"github.com/photoprism/photoprism/internal/config"
|
|
"github.com/photoprism/photoprism/internal/entity"
|
|
"github.com/photoprism/photoprism/internal/event"
|
|
"github.com/photoprism/photoprism/internal/mutex"
|
|
"github.com/photoprism/photoprism/internal/query"
|
|
"github.com/photoprism/photoprism/pkg/clean"
|
|
"github.com/photoprism/photoprism/pkg/fastwalk"
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
|
)
|
|
|
|
// CleanUp represents a worker that deletes unneeded data and files.
|
|
type CleanUp struct {
|
|
conf *config.Config
|
|
}
|
|
|
|
// NewCleanUp returns a new cleanup worker.
|
|
func NewCleanUp(conf *config.Config) *CleanUp {
|
|
instance := &CleanUp{
|
|
conf: conf,
|
|
}
|
|
|
|
return instance
|
|
}
|
|
|
|
// Start removes orphan index entries and thumbnails.
|
|
func (w *CleanUp) Start(opt CleanUpOptions) (thumbs int, orphans int, err error) {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
err = fmt.Errorf("cleanup: %s (panic)\nstack: %s", r, debug.Stack())
|
|
log.Error(err)
|
|
}
|
|
}()
|
|
|
|
if err = mutex.MainWorker.Start(); err != nil {
|
|
log.Warnf("cleanup: %s (start)", err.Error())
|
|
return thumbs, orphans, err
|
|
}
|
|
|
|
defer mutex.MainWorker.Stop()
|
|
|
|
if opt.Dry {
|
|
log.Infof("cleanup: dry run, nothing will actually be removed")
|
|
}
|
|
|
|
var fileHashes, thumbHashes query.HashMap
|
|
|
|
// Fetch existing media and thumb file hashes.
|
|
if fileHashes, err = query.FileHashMap(); err != nil {
|
|
return thumbs, orphans, err
|
|
} else if thumbHashes, err = query.ThumbHashMap(); err != nil {
|
|
return thumbs, orphans, err
|
|
}
|
|
|
|
// Thumbnails storage path.
|
|
thumbPath := w.conf.ThumbCachePath()
|
|
|
|
// Find and remove orphan thumbnail files.
|
|
if err := fastwalk.Walk(thumbPath, func(fileName string, info os.FileMode) error {
|
|
base := filepath.Base(fileName)
|
|
|
|
if info.IsDir() || strings.HasPrefix(base, ".") {
|
|
return nil
|
|
}
|
|
|
|
// Example: 01244519acf35c62a5fea7a5a7dcefdbec4fb2f5_3x3_resize.png
|
|
i := strings.Index(base, "_")
|
|
|
|
if i < 39 {
|
|
return nil
|
|
}
|
|
|
|
hash := base[:i]
|
|
logName := clean.Log(fs.RelName(fileName, thumbPath))
|
|
|
|
if ok := fileHashes[hash]; ok {
|
|
// Do nothing.
|
|
} else if ok := thumbHashes[hash]; ok {
|
|
// Do nothing.
|
|
} else if opt.Dry {
|
|
thumbs++
|
|
log.Debugf("cleanup: thumbnail %s would be removed", logName)
|
|
} else if err := os.Remove(fileName); err != nil {
|
|
log.Warnf("cleanup: %s in %s", err, logName)
|
|
} else {
|
|
thumbs++
|
|
log.Debugf("cleanup: removed thumbnail %s", logName)
|
|
}
|
|
|
|
return nil
|
|
}); err != nil {
|
|
return thumbs, orphans, err
|
|
}
|
|
|
|
// Find and remove orphan photo index entries.
|
|
photos, err := query.OrphanPhotos()
|
|
|
|
if err != nil {
|
|
return thumbs, orphans, err
|
|
}
|
|
|
|
var deleted []string
|
|
|
|
for _, p := range photos {
|
|
if mutex.MainWorker.Canceled() {
|
|
return thumbs, orphans, errors.New("cleanup canceled")
|
|
}
|
|
|
|
if opt.Dry {
|
|
orphans++
|
|
log.Infof("cleanup: orphan photo %s would be removed", clean.Log(p.PhotoUID))
|
|
continue
|
|
}
|
|
|
|
if err := Delete(p); err != nil {
|
|
log.Errorf("cleanup: %s (remove orphan photo)", err.Error())
|
|
} else {
|
|
orphans++
|
|
deleted = append(deleted, p.PhotoUID)
|
|
log.Debugf("cleanup: removed orphan photo %s", p.PhotoUID)
|
|
}
|
|
}
|
|
|
|
// Remove orphan index entries.
|
|
if opt.Dry {
|
|
if files, err := query.OrphanFiles(); err != nil {
|
|
log.Errorf("index: %s (find orphan files)", err)
|
|
} else if l := len(files); l > 0 {
|
|
log.Infof("index: found %s", english.Plural(l, "orphan file", "orphan files"))
|
|
} else {
|
|
log.Infof("index: found no orphan files")
|
|
}
|
|
} else {
|
|
if err := query.PurgeOrphans(); err != nil {
|
|
log.Errorf("index: %s (purge orphans)", err)
|
|
}
|
|
}
|
|
|
|
// Only update counts if anything was deleted.
|
|
if len(deleted) > 0 {
|
|
// Update precalculated photo and file counts.
|
|
if err := entity.UpdateCounts(); err != nil {
|
|
log.Warnf("index: %s (update counts)", err)
|
|
}
|
|
|
|
// Update album, subject, and label cover thumbs.
|
|
if err := query.UpdateCovers(); err != nil {
|
|
log.Warnf("index: %s (update covers)", err)
|
|
}
|
|
|
|
// Show success notification.
|
|
event.EntitiesDeleted("photos", deleted)
|
|
}
|
|
|
|
return thumbs, orphans, nil
|
|
}
|
|
|
|
// Cancel stops the current operation.
|
|
func (w *CleanUp) Cancel() {
|
|
mutex.MainWorker.Cancel()
|
|
}
|