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

179 lines
4.5 KiB
Go
Raw Normal View History

2020-02-01 17:58:45 +01:00
package scanner
import (
"container/list"
"fmt"
"io/ioutil"
"log"
"os"
"path"
2020-06-22 23:52:41 +02:00
"github.com/pkg/errors"
2020-02-01 17:58:45 +01:00
"github.com/viktorstrate/photoview/api/graphql/models"
2020-02-21 20:51:50 +01:00
"github.com/viktorstrate/photoview/api/graphql/notification"
"github.com/viktorstrate/photoview/api/utils"
"gorm.io/gorm"
2020-02-01 17:58:45 +01:00
)
func findAlbumsForUser(db *gorm.DB, user *models.User, album_cache *AlbumScannerCache) ([]*models.Album, []error) {
2020-02-21 20:51:50 +01:00
// Check if user directory exists on the file system
if _, err := os.Stat(user.RootPath); err != nil {
if os.IsNotExist(err) {
2020-06-22 23:52:41 +02:00
return nil, []error{errors.Errorf("Photo directory for user '%s' does not exist '%s'\n", user.Username, user.RootPath)}
} else {
2020-06-22 23:52:41 +02:00
return nil, []error{errors.Errorf("Could not read photo directory for user '%s': %s\n", user.Username, user.RootPath)}
}
}
2020-02-01 17:58:45 +01:00
type scanInfo struct {
path string
parentId *int
}
scanQueue := list.New()
scanQueue.PushBack(scanInfo{
path: user.RootPath,
parentId: nil,
})
2020-06-22 23:52:41 +02:00
userAlbums := make([]*models.Album, 0)
albumErrors := make([]error, 0)
// newPhotos := make([]*models.Photo, 0)
2020-02-01 17:58:45 +01:00
for scanQueue.Front() != nil {
albumInfo := scanQueue.Front().Value.(scanInfo)
scanQueue.Remove(scanQueue.Front())
albumPath := albumInfo.path
albumParentId := albumInfo.parentId
// Read path
dirContent, err := ioutil.ReadDir(albumPath)
if err != nil {
2020-06-22 23:52:41 +02:00
albumErrors = append(albumErrors, errors.Wrapf(err, "read directory (%s)", albumPath))
2020-02-23 11:59:57 +01:00
continue
2020-02-01 17:58:45 +01:00
}
2020-06-22 23:52:41 +02:00
tx, err := db.Begin()
2020-02-01 17:58:45 +01:00
if err != nil {
2020-06-22 23:52:41 +02:00
albumErrors = append(albumErrors, errors.Wrap(err, "begin database transaction"))
2020-02-23 11:59:57 +01:00
continue
2020-02-01 17:58:45 +01:00
}
log.Printf("Scanning directory: %s", albumPath)
// Make album if not exists
albumTitle := path.Base(albumPath)
2020-06-24 12:52:40 +02:00
_, err = tx.Exec("INSERT IGNORE INTO album (title, parent_album, owner_id, path, path_hash) VALUES (?, ?, ?, ?, MD5(path))", albumTitle, albumParentId, user.UserID, albumPath)
2020-02-01 17:58:45 +01:00
if err != nil {
2020-06-22 23:52:41 +02:00
albumErrors = append(albumErrors, errors.Wrap(err, "insert album into database"))
2020-02-01 17:58:45 +01:00
tx.Rollback()
2020-02-23 11:59:57 +01:00
continue
2020-02-01 17:58:45 +01:00
}
2020-06-24 12:52:40 +02:00
row := tx.QueryRow("SELECT * FROM album WHERE path_hash = MD5(?)", albumPath)
album, err := models.NewAlbumFromRow(row)
if err != nil {
2020-06-22 23:52:41 +02:00
albumErrors = append(albumErrors, errors.Wrapf(err, "get album from database (%s)", albumPath))
2020-02-01 17:58:45 +01:00
tx.Rollback()
2020-06-22 23:52:41 +02:00
continue
2020-02-01 17:58:45 +01:00
}
2020-06-22 23:52:41 +02:00
userAlbums = append(userAlbums, album)
2020-02-01 17:58:45 +01:00
2020-02-02 18:18:38 +01:00
// Commit album transaction
if err := tx.Commit(); err != nil {
2020-06-22 23:52:41 +02:00
albumErrors = append(albumErrors, errors.Wrap(err, "commit database transaction"))
continue
2020-02-01 17:58:45 +01:00
}
// Scan for sub-albums
for _, item := range dirContent {
subalbumPath := path.Join(albumPath, item.Name())
2020-03-07 15:34:32 +01:00
// Skip if directory is hidden
if path.Base(subalbumPath)[0:1] == "." {
continue
}
2020-06-22 23:52:41 +02:00
if item.IsDir() && directoryContainsPhotos(subalbumPath, album_cache) {
2020-02-01 17:58:45 +01:00
scanQueue.PushBack(scanInfo{
path: subalbumPath,
parentId: &album.AlbumID,
2020-02-01 17:58:45 +01:00
})
}
}
}
2020-06-22 23:52:41 +02:00
deleteErrors := deleteOldUserAlbums(db, userAlbums, user)
albumErrors = append(albumErrors, deleteErrors...)
2020-02-26 19:44:47 +01:00
2020-06-22 23:52:41 +02:00
return userAlbums, albumErrors
2020-02-01 17:58:45 +01:00
}
2020-06-22 23:52:41 +02:00
func directoryContainsPhotos(rootPath string, cache *AlbumScannerCache) bool {
2020-02-12 18:45:58 +01:00
2020-06-22 23:52:41 +02:00
if contains_image := cache.AlbumContainsPhotos(rootPath); contains_image != nil {
2020-02-12 18:45:58 +01:00
return *contains_image
}
2020-02-01 17:58:45 +01:00
scanQueue := list.New()
scanQueue.PushBack(rootPath)
2020-02-12 18:45:58 +01:00
scanned_directories := make([]string, 0)
2020-02-01 17:58:45 +01:00
for scanQueue.Front() != nil {
dirPath := scanQueue.Front().Value.(string)
scanQueue.Remove(scanQueue.Front())
2020-02-12 18:45:58 +01:00
scanned_directories = append(scanned_directories, dirPath)
2020-02-01 17:58:45 +01:00
dirContent, err := ioutil.ReadDir(dirPath)
if err != nil {
2020-02-23 11:59:57 +01:00
ScannerError("Could not read directory: %s\n", err.Error())
2020-02-01 17:58:45 +01:00
return false
}
for _, fileInfo := range dirContent {
filePath := path.Join(dirPath, fileInfo.Name())
if fileInfo.IsDir() {
scanQueue.PushBack(filePath)
} else {
if isPathMedia(filePath, cache) {
2020-06-22 23:52:41 +02:00
cache.InsertAlbumPaths(dirPath, rootPath, true)
2020-02-01 17:58:45 +01:00
return true
}
}
}
}
2020-02-12 18:45:58 +01:00
for _, scanned_path := range scanned_directories {
2020-06-22 23:52:41 +02:00
cache.InsertAlbumPath(scanned_path, false)
2020-02-12 18:45:58 +01:00
}
2020-02-01 17:58:45 +01:00
return false
}
2020-02-12 18:10:52 +01:00
2020-02-23 11:59:57 +01:00
func ScannerError(format string, args ...interface{}) {
message := fmt.Sprintf(format, args...)
log.Printf("ERROR: %s", message)
notification.BroadcastNotification(&models.Notification{
Key: utils.GenerateToken(),
Type: models.NotificationTypeMessage,
Header: "Scanner error",
Content: message,
Negative: true,
})
}
2020-02-23 12:43:45 +01:00
func PhotoCache() string {
photoCache := os.Getenv("PHOTO_CACHE")
if photoCache == "" {
photoCache = "./photo_cache"
}
return photoCache
}