604849e92c
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>
161 lines
4.4 KiB
Go
161 lines
4.4 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 cameraMutex = sync.Mutex{}
|
|
|
|
// Cameras represents a list of cameras.
|
|
type Cameras []Camera
|
|
|
|
// Camera model and make (as extracted from UpdateExif metadata)
|
|
type Camera struct {
|
|
ID uint `gorm:"primary_key" json:"ID" yaml:"ID"`
|
|
CameraSlug string `gorm:"type:VARBINARY(160);unique_index;" json:"Slug" yaml:"-"`
|
|
CameraName string `gorm:"type:VARCHAR(160);" json:"Name" yaml:"Name"`
|
|
CameraMake string `gorm:"type:VARCHAR(160);" json:"Make" yaml:"Make,omitempty"`
|
|
CameraModel string `gorm:"type:VARCHAR(160);" json:"Model" yaml:"Model,omitempty"`
|
|
CameraType string `gorm:"type:VARCHAR(100);" json:"Type,omitempty" yaml:"Type,omitempty"`
|
|
CameraDescription string `gorm:"type:VARCHAR(2048);" json:"Description,omitempty" yaml:"Description,omitempty"`
|
|
CameraNotes 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 (Camera) TableName() string {
|
|
return "cameras"
|
|
}
|
|
|
|
var UnknownCamera = Camera{
|
|
CameraSlug: UnknownID,
|
|
CameraName: "Unknown",
|
|
CameraMake: "",
|
|
CameraModel: "Unknown",
|
|
}
|
|
|
|
// CreateUnknownCamera initializes the database with an unknown camera if not exists
|
|
func CreateUnknownCamera() {
|
|
UnknownCamera = *FirstOrCreateCamera(&UnknownCamera)
|
|
}
|
|
|
|
// NewCamera creates a camera entity from a model name and a make name.
|
|
func NewCamera(modelName string, makeName string) *Camera {
|
|
modelName = strings.TrimSpace(modelName)
|
|
makeName = strings.TrimSpace(makeName)
|
|
|
|
if modelName == "" && makeName == "" {
|
|
return &UnknownCamera
|
|
} else if strings.HasPrefix(modelName, makeName) {
|
|
modelName = strings.TrimSpace(modelName[len(makeName):])
|
|
}
|
|
|
|
// Normalize make name.
|
|
if n, ok := CameraMakes[makeName]; ok {
|
|
makeName = n
|
|
}
|
|
|
|
// Normalize model name.
|
|
if n, ok := CameraModels[modelName]; ok {
|
|
modelName = 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)
|
|
}
|
|
|
|
cameraName := strings.Join(name, " ")
|
|
|
|
result := &Camera{
|
|
CameraSlug: txt.Slug(cameraName),
|
|
CameraName: txt.Clip(cameraName, txt.ClipName),
|
|
CameraMake: txt.Clip(makeName, txt.ClipName),
|
|
CameraModel: txt.Clip(modelName, txt.ClipName),
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// Create inserts a new row to the database.
|
|
func (m *Camera) Create() error {
|
|
cameraMutex.Lock()
|
|
defer cameraMutex.Unlock()
|
|
|
|
return Db().Create(m).Error
|
|
}
|
|
|
|
// FirstOrCreateCamera returns the existing row, inserts a new row or nil in case of errors.
|
|
func FirstOrCreateCamera(m *Camera) *Camera {
|
|
if m.CameraSlug == "" {
|
|
return &UnknownCamera
|
|
}
|
|
|
|
if cacheData, ok := cameraCache.Get(m.CameraSlug); ok {
|
|
log.Tracef("camera: cache hit for %s", m.CameraSlug)
|
|
|
|
return cacheData.(*Camera)
|
|
}
|
|
|
|
result := Camera{}
|
|
|
|
if res := Db().Where("camera_slug = ?", m.CameraSlug).First(&result); res.Error == nil {
|
|
cameraCache.SetDefault(m.CameraSlug, &result)
|
|
return &result
|
|
} else if err := m.Create(); err == nil {
|
|
if !m.Unknown() {
|
|
event.EntitiesCreated("cameras", []*Camera{m})
|
|
|
|
event.Publish("count.cameras", event.Data{
|
|
"count": 1,
|
|
})
|
|
}
|
|
|
|
cameraCache.SetDefault(m.CameraSlug, m)
|
|
|
|
return m
|
|
} else if res := Db().Where("camera_slug = ?", m.CameraSlug).First(&result); res.Error == nil {
|
|
cameraCache.SetDefault(m.CameraSlug, &result)
|
|
return &result
|
|
} else {
|
|
log.Errorf("camera: %s (create %s)", err.Error(), clean.Log(m.String()))
|
|
}
|
|
|
|
return &UnknownCamera
|
|
}
|
|
|
|
// String returns an identifier that can be used in logs.
|
|
func (m *Camera) String() string {
|
|
return clean.Log(m.CameraName)
|
|
}
|
|
|
|
// Scanner checks whether the model appears to be a scanner.
|
|
func (m *Camera) Scanner() bool {
|
|
if m.CameraSlug == "" {
|
|
return false
|
|
}
|
|
|
|
return strings.Contains(m.CameraSlug, "scan")
|
|
}
|
|
|
|
// Unknown returns true if the camera is not a known make or model.
|
|
func (m *Camera) Unknown() bool {
|
|
return m.CameraSlug == "" || m.CameraSlug == UnknownCamera.CameraSlug
|
|
}
|