1
Fork 0
photoview/api/scanner/scanner_album.go

172 lines
4.5 KiB
Go
Raw Normal View History

package scanner
import (
"fmt"
"io/ioutil"
"path"
"time"
2020-12-17 22:51:43 +01:00
"github.com/photoview/photoview/api/graphql/models"
"github.com/photoview/photoview/api/graphql/notification"
2021-02-15 17:35:28 +01:00
"github.com/photoview/photoview/api/scanner/face_detection"
2020-12-17 22:51:43 +01:00
"github.com/photoview/photoview/api/utils"
2020-11-23 19:59:01 +01:00
"github.com/pkg/errors"
"gorm.io/gorm"
)
2020-12-22 01:14:43 +01:00
func NewRootAlbum(db *gorm.DB, rootPath string, owner *models.User) (*models.Album, error) {
owners := []models.User{
*owner,
}
2020-12-31 00:37:11 +01:00
var matchedAlbums []models.Album
2021-01-17 12:45:23 +01:00
if err := db.Where("path_hash = ?", models.MD5Hash(rootPath)).Find(&matchedAlbums).Error; err != nil {
2020-12-22 01:14:43 +01:00
return nil, err
}
2020-12-31 00:37:11 +01:00
if len(matchedAlbums) > 0 {
album := matchedAlbums[0]
if err := db.Model(&owner).Association("Albums").Append(&album); err != nil {
return nil, errors.Wrap(err, "failed to add owner to already existing album")
}
return &album, nil
} else {
album := models.Album{
Title: path.Base(rootPath),
Path: rootPath,
Owners: owners,
}
if err := db.Create(&album).Error; err != nil {
return nil, err
}
return &album, nil
}
2020-12-22 01:14:43 +01:00
}
func scanAlbum(album *models.Album, cache *AlbumScannerCache, db *gorm.DB) {
album_notify_key := utils.GenerateToken()
notifyThrottle := utils.NewThrottle(500 * time.Millisecond)
notifyThrottle.Trigger(nil)
2020-06-22 23:52:41 +02:00
// Scan for photos
2021-02-15 17:35:28 +01:00
albumMedia, err := findMediaForAlbum(album, cache, db, func(photo *models.Media, newPhoto bool) {
if newPhoto {
notifyThrottle.Trigger(func() {
notification.BroadcastNotification(&models.Notification{
Key: album_notify_key,
Type: models.NotificationTypeMessage,
Header: fmt.Sprintf("Found new media in album '%s'", album.Title),
Content: fmt.Sprintf("Found %s", photo.Path),
})
})
}
2020-06-22 23:52:41 +02:00
})
if err != nil {
ScannerError("Failed to find media for album (%s): %s", album.Path, err)
2020-06-22 23:52:41 +02:00
}
album_has_changes := false
2021-02-15 17:35:28 +01:00
for count, media := range albumMedia {
2020-11-23 19:59:01 +01:00
// tx, err := db.Begin()
2020-11-30 21:29:49 +01:00
transactionError := db.Transaction(func(tx *gorm.DB) error {
2021-02-15 17:35:28 +01:00
processing_was_needed, err := ProcessMedia(tx, media)
2020-11-23 19:59:01 +01:00
if err != nil {
2021-02-15 17:35:28 +01:00
return errors.Wrapf(err, "failed to process photo (%s)", media.Path)
2020-11-23 19:59:01 +01:00
}
2020-06-22 23:52:41 +02:00
2020-11-23 19:59:01 +01:00
if processing_was_needed {
album_has_changes = true
2021-02-15 17:35:28 +01:00
progress := float64(count) / float64(len(albumMedia)) * 100.0
2020-11-23 19:59:01 +01:00
notification.BroadcastNotification(&models.Notification{
Key: album_notify_key,
Type: models.NotificationTypeProgress,
Header: fmt.Sprintf("Processing media for album '%s'", album.Title),
2021-02-15 17:35:28 +01:00
Content: fmt.Sprintf("Processed media at %s", media.Path),
2020-11-23 19:59:01 +01:00
Progress: &progress,
})
2021-02-15 17:35:28 +01:00
if media.Type == models.MediaTypePhoto {
go func() {
if err := face_detection.GlobalFaceDetector.DetectFaces(media); err != nil {
ScannerError("Error detecting faces in image (%s): %s", media.Path, err)
}
}()
}
2020-11-23 19:59:01 +01:00
}
2020-11-23 19:59:01 +01:00
return nil
})
2020-11-30 21:29:49 +01:00
if transactionError != nil {
ScannerError("Failed to begin database transaction: %s", transactionError)
}
}
2021-02-15 17:35:28 +01:00
cleanup_errors := CleanupMedia(db, album.ID, albumMedia)
for _, err := range cleanup_errors {
ScannerError("Failed to delete old media: %s", err)
}
if album_has_changes {
timeoutDelay := 2000
notification.BroadcastNotification(&models.Notification{
Key: album_notify_key,
Type: models.NotificationTypeMessage,
Positive: true,
Header: fmt.Sprintf("Done processing media for album '%s'", album.Title),
Content: fmt.Sprintf("All media have been processed"),
Timeout: &timeoutDelay,
})
2020-06-22 23:52:41 +02:00
}
}
func findMediaForAlbum(album *models.Album, cache *AlbumScannerCache, db *gorm.DB, onScanPhoto func(photo *models.Media, newPhoto bool)) ([]*models.Media, error) {
albumPhotos := make([]*models.Media, 0)
dirContent, err := ioutil.ReadDir(album.Path)
if err != nil {
return nil, err
}
for _, item := range dirContent {
photoPath := path.Join(album.Path, item.Name())
if !item.IsDir() && isPathMedia(photoPath, cache) {
// Skip the JPEGs that are compressed version of raw files
counterpartFile := scanForRawCounterpartFile(photoPath)
if counterpartFile != nil {
continue
}
err := db.Transaction(func(tx *gorm.DB) error {
2021-02-15 17:35:28 +01:00
media, isNewMedia, err := ScanMedia(tx, photoPath, album.ID, cache)
2020-11-23 19:59:01 +01:00
if err != nil {
return errors.Wrapf(err, "Scanning media error (%s)", photoPath)
}
2021-02-15 17:35:28 +01:00
onScanPhoto(media, isNewMedia)
2021-02-15 17:35:28 +01:00
albumPhotos = append(albumPhotos, media)
2020-11-23 19:59:01 +01:00
return nil
})
if err != nil {
2020-12-14 10:41:00 +01:00
ScannerError("Error scanning media for album (%d): %s\n", album.ID, err)
continue
}
}
}
return albumPhotos, nil
}