photoprism/mediafile.go
2018-02-27 19:04:48 +01:00

285 lines
5.1 KiB
Go

package photoprism
import (
"path/filepath"
"encoding/hex"
"github.com/brett-lempereur/ish"
"net/http"
"os"
"log"
"strings"
"time"
"github.com/djherbis/times"
"github.com/steakknife/hamming"
)
const (
FileTypeOther = ""
FileTypeYaml = "yml"
FileTypeJpeg = "jpg"
FileTypeRaw = "raw"
FileTypeXmp = "xmp"
FileTypeAae = "aae"
FileTypeMovie = "mov"
)
const (
MimeTypeJpeg = "image/jpeg"
)
var FileExtensions = map[string]string {
".crw": FileTypeRaw,
".cr2": FileTypeRaw,
".nef": FileTypeRaw,
".arw": FileTypeRaw,
".dng": FileTypeRaw,
".mov": FileTypeMovie,
".avi": FileTypeMovie,
".yml": FileTypeYaml,
".jpg": FileTypeJpeg,
".jpeg": FileTypeJpeg,
".xmp": FileTypeXmp,
".aae": FileTypeAae,
}
type MediaFile struct {
filename string
dateCreated time.Time
hash string
fileType string
mimeType string
perceptualHash string
tags []string
exifData *ExifData
}
func NewMediaFile(filename string) *MediaFile {
instance := &MediaFile{
filename: filename,
fileType: FileTypeOther,
}
return instance
}
func (m *MediaFile) GetDateCreated() time.Time {
if !m.dateCreated.IsZero() {
return m.dateCreated
}
info, err := m.GetExifData()
if err == nil {
m.dateCreated = info.DateTime
return info.DateTime
}
t, err := times.Stat(m.GetFilename())
if err != nil {
log.Fatal(err.Error())
}
if t.HasBirthTime() {
m.dateCreated = t.BirthTime()
return t.BirthTime()
}
m.dateCreated = t.ModTime()
return t.ModTime()
}
func (m *MediaFile) GetCameraModel () string {
info, err := m.GetExifData()
var result string
if err == nil {
result = info.CameraModel
}
return result
}
func (m *MediaFile) GetCanonicalName() string {
dateCreated := m.GetDateCreated().UTC()
//cameraModel := strings.Replace(m.GetCameraModel(), " ", "_", -1)
result := dateCreated.Format("20060102_150405_") + strings.ToUpper(m.GetHash()[:12])
/* if cameraModel != "" {
result = result + "_" + cameraModel
} */
return result
}
func (m *MediaFile) GetPerceptualHash() (string, error) {
if m.perceptualHash != "" {
return m.perceptualHash, nil
}
hasher := ish.NewDifferenceHash(8, 8)
img, _, err := ish.LoadFile(m.GetFilename())
if err != nil {
return "", err
}
dh, err := hasher.Hash(img)
if err != nil {
return "", err
}
m.perceptualHash = hex.EncodeToString(dh)
return m.perceptualHash, nil
}
func (m *MediaFile) GetPerceptualDistance(perceptualHash string) (int, error) {
var hash1, hash2 []byte
if imageHash, err := m.GetPerceptualHash(); err != nil {
return -1, err
} else {
if decoded, err := hex.DecodeString(imageHash); err != nil {
return -1, err
} else {
hash1 = decoded
}
}
if decoded, err := hex.DecodeString(perceptualHash); err != nil {
return -1, err
} else {
hash2 = decoded
}
result := hamming.Bytes(hash1, hash2)
return result, nil
}
func (m *MediaFile) GetHash() string {
if len(m.hash) == 0 {
m.hash = fileHash(m.GetFilename())
}
return m.hash
}
func (m *MediaFile) GetRelatedFiles() (result []*MediaFile, masterFile *MediaFile, err error) {
extension := m.GetExtension()
baseFilename := m.filename[0:len(m.filename)-len(extension)]
matches, err := filepath.Glob(baseFilename + "*")
if err != nil {
return result, nil, err
}
for _, filename := range matches {
resultFile := NewMediaFile(filename)
if masterFile == nil && resultFile.IsJpeg() {
masterFile = resultFile
} else if resultFile.IsRaw() {
masterFile = resultFile
}
result = append(result, resultFile)
}
return result, masterFile, nil
}
func (m *MediaFile) GetFilename() string {
return m.filename
}
func (m *MediaFile) SetFilename(filename string) {
m.filename = filename
}
func (m *MediaFile) GetMimeType() string {
if m.mimeType != "" {
return m.mimeType
}
handle, err := m.openFile()
if err != nil {
log.Println("Error: Could not open file to determine mime type")
return ""
}
defer handle.Close()
// Only the first 512 bytes are used to sniff the content type.
buffer := make([]byte, 512)
_, err = handle.Read(buffer)
if err != nil {
log.Println("Error: Could not read file to determine mime type: " + m.GetFilename())
return ""
}
m.mimeType = http.DetectContentType(buffer)
return m.mimeType
}
func (m *MediaFile) openFile() (*os.File, error) {
if handle, err := os.Open(m.filename); err == nil {
return handle, nil
} else {
log.Println(err.Error())
return nil, err
}
}
func (m *MediaFile) Exists() bool {
return fileExists(m.GetFilename())
}
func (m *MediaFile) Remove() error {
return os.Remove(m.GetFilename())
}
func (m *MediaFile) Move(newFilename string) error {
if err := os.Rename(m.filename, newFilename); err != nil {
return err
}
m.filename = newFilename
return nil
}
func (m *MediaFile) GetExtension() string {
return strings.ToLower(filepath.Ext(m.filename))
}
func (m *MediaFile) IsJpeg() bool {
return m.GetMimeType() == MimeTypeJpeg
}
func (m *MediaFile) HasType(typeString string) bool {
if typeString == FileTypeJpeg {
return m.IsJpeg()
}
return FileExtensions[m.GetExtension()] == typeString
}
func (m *MediaFile) IsRaw() bool {
return m.HasType(FileTypeRaw)
}
func (m *MediaFile) IsPhoto() bool {
return m.IsJpeg() || m.IsRaw()
}