Files
photoprism/internal/entity/photo_optimize.go
Ömer Duran 1e00d1f52e UX: Add batch edit dialog and API endpoints #271 #5324
Signed-off-by: Michael Mayer <michael@photoprism.app>
Co-authored-by: Michael Mayer <michael@photoprism.app>
Co-authored-by: graciousgrey <theresagresch@gmail.com>
2025-11-19 11:20:34 +01:00

66 lines
1.7 KiB
Go

package entity
import (
"errors"
"reflect"
)
// Optimize updates picture metadata, enriching titles, keywords, and locations according to the supplied flags.
func (m *Photo) Optimize(mergeMeta, mergeUuid, estimateLocation, force bool) (updated bool, merged Photos, err error) {
if !m.HasID() {
return false, merged, errors.New("photo: cannot maintain, id is empty")
}
// Keep a snapshot so we can detect whether anything changed.
current := *m
if m.HasLatLng() && !m.HasLocation() {
m.UpdateLocation()
}
if original, photos, mergeErr := m.Merge(mergeMeta, mergeUuid); mergeErr != nil {
return updated, merged, mergeErr
} else if len(photos) > 0 && original.ID == m.ID {
merged = photos
} else if len(photos) > 0 && original.ID != m.ID {
return false, photos, nil
}
// Estimate the location if it is unknown and this feature is enabled.
if estimateLocation && SrcPriority[m.PlaceSrc] <= SrcPriority[SrcEstimate] {
m.EstimateLocation(force)
}
// Get image classification labels.
labels := m.ClassifyLabels()
m.UpdateDateFields()
if updateErr := m.GenerateTitle(labels); updateErr != nil {
log.Info(updateErr)
}
if indexErr := m.IndexKeywords(); indexErr != nil {
log.Errorf("photo: %s", indexErr.Error())
}
m.PhotoQuality = m.QualityScore()
checked := Now()
// Skip persistence when nothing changed besides the indexing timestamps.
if reflect.DeepEqual(*m, current) {
return false, merged, m.Updates(Values{
"IndexedAt": m.Indexed(),
"CheckedAt": &checked},
)
}
// Ensure IndexedAt remains set even when CheckedAt gets refreshed.
m.IndexedAt = m.Indexed()
m.CheckedAt = &checked
// Persist the updated metadata to the database.
return true, merged, m.Save()
}