2020-05-26 19:27:29 +02:00
|
|
|
package workers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2020-07-13 10:41:45 +02:00
|
|
|
"fmt"
|
2020-05-26 19:27:29 +02:00
|
|
|
"runtime"
|
2020-07-13 10:41:45 +02:00
|
|
|
"runtime/debug"
|
2020-05-26 19:27:29 +02:00
|
|
|
"time"
|
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
"github.com/dustin/go-humanize/english"
|
|
|
|
|
2020-05-26 19:27:29 +02:00
|
|
|
"github.com/photoprism/photoprism/internal/config"
|
|
|
|
"github.com/photoprism/photoprism/internal/entity"
|
|
|
|
"github.com/photoprism/photoprism/internal/mutex"
|
2020-05-30 14:52:47 +02:00
|
|
|
"github.com/photoprism/photoprism/internal/photoprism"
|
2020-05-26 19:27:29 +02:00
|
|
|
"github.com/photoprism/photoprism/internal/query"
|
|
|
|
)
|
|
|
|
|
2023-10-15 11:02:15 +02:00
|
|
|
// Meta represents a background index and metadata optimization worker.
|
2020-06-29 13:35:38 +02:00
|
|
|
type Meta struct {
|
2023-03-08 12:42:57 +01:00
|
|
|
conf *config.Config
|
|
|
|
lastRun time.Time
|
2020-05-26 19:27:29 +02:00
|
|
|
}
|
|
|
|
|
2021-08-14 14:24:48 +02:00
|
|
|
// NewMeta returns a new Meta worker.
|
2020-06-29 13:35:38 +02:00
|
|
|
func NewMeta(conf *config.Config) *Meta {
|
|
|
|
return &Meta{conf: conf}
|
2020-05-26 19:27:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// originalsPath returns the original media files path as string.
|
2023-03-08 12:42:57 +01:00
|
|
|
func (w *Meta) originalsPath() string {
|
|
|
|
return w.conf.OriginalsPath()
|
2020-05-26 19:27:29 +02:00
|
|
|
}
|
|
|
|
|
2021-08-14 14:24:48 +02:00
|
|
|
// Start metadata optimization routine.
|
2023-03-08 12:42:57 +01:00
|
|
|
func (w *Meta) Start(delay, interval time.Duration, force bool) (err error) {
|
2020-07-13 10:41:45 +02:00
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
2023-03-08 12:42:57 +01:00
|
|
|
err = fmt.Errorf("index: %s (worker panic)\nstack: %s", r, debug.Stack())
|
2020-07-13 10:41:45 +02:00
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
if err = mutex.MetaWorker.Start(); err != nil {
|
2020-05-26 19:27:29 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-07-13 10:41:45 +02:00
|
|
|
defer mutex.MetaWorker.Stop()
|
2020-05-26 19:27:29 +02:00
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
// Check time when worker was last executed.
|
|
|
|
updateIndex := force || w.lastRun.Before(time.Now().Add(-1*entity.IndexUpdateInterval))
|
2021-09-29 20:09:34 +02:00
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
// Run faces worker if needed.
|
|
|
|
if updateIndex || entity.UpdateFaces.Load() {
|
|
|
|
log.Debugf("index: running face recognition")
|
|
|
|
if faces := photoprism.NewFaces(w.conf); faces.Disabled() {
|
|
|
|
log.Debugf("index: skipping face recognition")
|
|
|
|
} else if err := faces.Start(photoprism.FacesOptions{}); err != nil {
|
|
|
|
log.Warn(err)
|
|
|
|
}
|
2021-09-29 20:09:34 +02:00
|
|
|
}
|
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
// Refresh index metadata.
|
|
|
|
log.Debugf("index: updating metadata")
|
2020-06-29 13:16:55 +02:00
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
start := time.Now()
|
|
|
|
settings := w.conf.Settings()
|
2020-05-26 19:27:29 +02:00
|
|
|
done := make(map[string]bool)
|
2023-03-08 12:42:57 +01:00
|
|
|
limit := 1000
|
2020-05-26 19:27:29 +02:00
|
|
|
offset := 0
|
2020-06-29 13:16:55 +02:00
|
|
|
optimized := 0
|
2020-05-26 19:27:29 +02:00
|
|
|
|
|
|
|
for {
|
2021-11-20 19:14:00 +01:00
|
|
|
photos, err := query.PhotosMetadataUpdate(limit, offset, delay, interval)
|
2020-05-26 19:27:29 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(photos) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, photo := range photos {
|
2020-06-29 13:35:38 +02:00
|
|
|
if mutex.MetaWorker.Canceled() {
|
2023-10-15 11:02:15 +02:00
|
|
|
return errors.New("index: metadata optimization canceled")
|
2020-05-26 19:27:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if done[photo.PhotoUID] {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
done[photo.PhotoUID] = true
|
|
|
|
|
2021-11-18 03:47:25 +01:00
|
|
|
updated, merged, err := photo.Optimize(settings.StackMeta(), settings.StackUUID(), settings.Features.Estimates, force)
|
2020-12-09 22:15:10 +01:00
|
|
|
|
|
|
|
if err != nil {
|
2023-10-15 11:02:15 +02:00
|
|
|
log.Errorf("index: %s in optimization worker", err)
|
2020-06-29 13:16:55 +02:00
|
|
|
} else if updated {
|
|
|
|
optimized++
|
2023-03-08 12:42:57 +01:00
|
|
|
log.Debugf("index: updated photo %s", photo.String())
|
2020-06-29 13:16:55 +02:00
|
|
|
}
|
2020-12-09 22:15:10 +01:00
|
|
|
|
2021-08-14 14:24:48 +02:00
|
|
|
for _, p := range merged {
|
2023-03-08 12:42:57 +01:00
|
|
|
log.Infof("index: merged %s", p.PhotoUID)
|
2021-08-14 14:24:48 +02:00
|
|
|
done[p.PhotoUID] = true
|
2020-12-09 22:15:10 +01:00
|
|
|
}
|
2020-05-26 19:27:29 +02:00
|
|
|
}
|
|
|
|
|
2020-06-29 13:35:38 +02:00
|
|
|
if mutex.MetaWorker.Canceled() {
|
2023-10-15 11:02:15 +02:00
|
|
|
return errors.New("index: optimization canceled")
|
2020-05-26 19:27:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
offset += limit
|
|
|
|
}
|
|
|
|
|
2020-06-29 13:16:55 +02:00
|
|
|
if optimized > 0 {
|
2023-03-08 12:42:57 +01:00
|
|
|
log.Infof("index: updated %s [%s]", english.Plural(optimized, "photo", "photos"), time.Since(start))
|
|
|
|
updateIndex = true
|
2020-05-26 19:27:29 +02:00
|
|
|
}
|
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
// Only update index if necessary.
|
|
|
|
if updateIndex {
|
|
|
|
// Set photo quality scores to -1 if files are missing.
|
|
|
|
if err = query.FlagHiddenPhotos(); err != nil {
|
2023-10-15 11:02:15 +02:00
|
|
|
log.Warnf("index: %s in optimization worker", err)
|
2023-03-08 12:42:57 +01:00
|
|
|
}
|
2020-05-26 19:27:29 +02:00
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
// Run moments worker.
|
|
|
|
if moments := photoprism.NewMoments(w.conf); moments == nil {
|
|
|
|
log.Errorf("index: failed updating moments")
|
|
|
|
} else if err = moments.Start(); err != nil {
|
2023-10-15 11:02:15 +02:00
|
|
|
log.Warnf("moments: %s in optimization worker", err)
|
2023-03-08 12:42:57 +01:00
|
|
|
}
|
2020-05-30 14:52:47 +02:00
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
// Update precalculated photo and file counts.
|
|
|
|
if err = entity.UpdateCounts(); err != nil {
|
2023-10-15 11:02:15 +02:00
|
|
|
log.Warnf("index: %s in optimization worker", err)
|
2023-03-08 12:42:57 +01:00
|
|
|
}
|
2021-09-23 16:06:59 +02:00
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
// Update album, subject, and label cover thumbs.
|
|
|
|
if err = query.UpdateCovers(); err != nil {
|
2023-10-15 11:02:15 +02:00
|
|
|
log.Warnf("index: %s in optimization worker", err)
|
2023-03-08 12:42:57 +01:00
|
|
|
}
|
2021-08-14 14:24:48 +02:00
|
|
|
}
|
|
|
|
|
2023-03-08 12:42:57 +01:00
|
|
|
// Update time when worker was last executed.
|
|
|
|
w.lastRun = entity.TimeStamp()
|
|
|
|
|
2021-08-14 14:24:48 +02:00
|
|
|
// Run garbage collection.
|
2020-05-26 19:27:29 +02:00
|
|
|
runtime.GC()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|