photoprism/internal/entity/lens.go
Michael Mayer 09ad17d10a Cards View: Improve camera and lens information #2040 #3077 #3816
Signed-off-by: Michael Mayer <michael@photoprism.app>
2023-10-13 11:35:43 +02:00

151 lines
4.1 KiB
Go

package entity
import (
"strings"
"sync"
"time"
"github.com/photoprism/photoprism/internal/event"
"github.com/photoprism/photoprism/pkg/clean"
"github.com/photoprism/photoprism/pkg/txt"
)
var lensMutex = sync.Mutex{}
// Lenses represents a list of lenses.
type Lenses []Lens
// Lens represents camera lens (as extracted from UpdateExif metadata)
type Lens struct {
ID uint `gorm:"primary_key" json:"ID" yaml:"ID"`
LensSlug string `gorm:"type:VARBINARY(160);unique_index;" json:"Slug" yaml:"Slug,omitempty"`
LensName string `gorm:"type:VARCHAR(160);" json:"Name" yaml:"Name"`
LensMake string `gorm:"type:VARCHAR(160);" json:"Make" yaml:"Make,omitempty"`
LensModel string `gorm:"type:VARCHAR(160);" json:"Model" yaml:"Model,omitempty"`
LensType string `gorm:"type:VARCHAR(100);" json:"Type" yaml:"Type,omitempty"`
LensDescription string `gorm:"type:VARCHAR(2048);" json:"Description,omitempty" yaml:"Description,omitempty"`
LensNotes string `gorm:"type:VARCHAR(1024);" json:"Notes,omitempty" yaml:"Notes,omitempty"`
CreatedAt time.Time `json:"-" yaml:"-"`
UpdatedAt time.Time `json:"-" yaml:"-"`
DeletedAt *time.Time `sql:"index" json:"-" yaml:"-"`
}
// TableName returns the entity table name.
func (Lens) TableName() string {
return "lenses"
}
var UnknownLens = Lens{
LensSlug: UnknownID,
LensName: "Unknown",
LensMake: "",
LensModel: "Unknown",
}
// CreateUnknownLens initializes the database with an unknown lens if not exists
func CreateUnknownLens() {
UnknownLens = *FirstOrCreateLens(&UnknownLens)
}
// NewLens creates a new camera lens entity from make and model names.
func NewLens(makeName string, modelName string) *Lens {
makeName = strings.TrimSpace(makeName)
modelName = strings.TrimSpace(modelName)
if modelName == "" && makeName == "" {
return &UnknownLens
} else if strings.HasPrefix(modelName, makeName) {
modelName = strings.TrimSpace(modelName[len(makeName):])
}
// Normalize make name.
if n, ok := CameraMakes[makeName]; ok {
makeName = n
}
// Remove duplicate make from model name.
if strings.HasPrefix(modelName, makeName) {
modelName = strings.TrimSpace(modelName[len(makeName):])
}
// Remove ignored substrings from model name.
modelName = LensModelIgnore.ReplaceAllString(modelName, " ")
var name []string
if makeName != "" {
name = append(name, makeName)
}
if modelName != "" {
name = append(name, modelName)
}
lensName := strings.Join(name, " ")
result := &Lens{
LensSlug: txt.Slug(lensName),
LensName: txt.Clip(lensName, txt.ClipName),
LensMake: txt.Clip(makeName, txt.ClipName),
LensModel: txt.Clip(modelName, txt.ClipName),
}
return result
}
// Create inserts a new row to the database.
func (m *Lens) Create() error {
lensMutex.Lock()
defer lensMutex.Unlock()
return Db().Create(m).Error
}
// FirstOrCreateLens returns the existing row, inserts a new row or nil in case of errors.
func FirstOrCreateLens(m *Lens) *Lens {
if m.LensSlug == "" {
return &UnknownLens
}
if cacheData, ok := lensCache.Get(m.LensSlug); ok {
log.Tracef("lens: cache hit for %s", m.LensSlug)
return cacheData.(*Lens)
}
result := Lens{}
if res := Db().Where("lens_slug = ?", m.LensSlug).First(&result); res.Error == nil {
lensCache.SetDefault(m.LensSlug, &result)
return &result
} else if err := m.Create(); err == nil {
if !m.Unknown() {
event.EntitiesCreated("lenses", []*Lens{m})
event.Publish("count.lenses", event.Data{
"count": 1,
})
}
lensCache.SetDefault(m.LensSlug, m)
return m
} else if res := Db().Where("lens_slug = ?", m.LensSlug).First(&result); res.Error == nil {
lensCache.SetDefault(m.LensSlug, &result)
return &result
} else {
log.Errorf("lens: %s (create %s)", err.Error(), clean.Log(m.String()))
}
return &UnknownLens
}
// String returns an identifier that can be used in logs.
func (m *Lens) String() string {
return clean.Log(m.LensName)
}
// Unknown returns true if the lens is not a known make or model.
func (m *Lens) Unknown() bool {
return m.LensSlug == "" || m.LensSlug == UnknownLens.LensSlug
}