photoprism/internal/entity/lens.go
Michael Mayer 604849e92c Search: Include RAW files in results by default #2040
With these changes the size and type of the RAW file as well as other
details can be displayed in the Cards View. This also improves the
indexing of camera and lens metadata.

Signed-off-by: Michael Mayer <michael@photoprism.app>
2023-10-06 02:22:48 +02:00

148 lines
3.9 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 lens in database
func NewLens(modelName string, makeName string) *Lens {
modelName = strings.TrimSpace(modelName)
makeName = strings.TrimSpace(makeName)
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
}
if strings.HasPrefix(modelName, makeName) {
modelName = strings.TrimSpace(modelName[len(makeName):])
}
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
}