2020-02-05 14:51:46 +01:00
|
|
|
package resolvers
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
2020-12-17 22:51:43 +01:00
|
|
|
api "github.com/photoview/photoview/api/graphql"
|
|
|
|
"github.com/photoview/photoview/api/graphql/auth"
|
|
|
|
"github.com/photoview/photoview/api/graphql/models"
|
|
|
|
"github.com/photoview/photoview/api/scanner"
|
2020-07-10 18:52:18 +02:00
|
|
|
"github.com/pkg/errors"
|
2020-11-28 17:31:19 +01:00
|
|
|
"gorm.io/gorm"
|
2020-12-20 17:10:00 +01:00
|
|
|
"gorm.io/gorm/clause"
|
2020-02-05 14:51:46 +01:00
|
|
|
)
|
|
|
|
|
2020-07-10 14:26:19 +02:00
|
|
|
func (r *queryResolver) MyMedia(ctx context.Context, filter *models.Filter) ([]*models.Media, error) {
|
2020-02-09 14:21:53 +01:00
|
|
|
user := auth.UserFromContext(ctx)
|
|
|
|
if user == nil {
|
|
|
|
return nil, errors.New("unauthorized")
|
|
|
|
}
|
|
|
|
|
2020-12-22 01:14:43 +01:00
|
|
|
if err := user.FillAlbums(r.Database); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
userAlbumIDs := make([]int, len(user.Albums))
|
|
|
|
for i, album := range user.Albums {
|
|
|
|
userAlbumIDs[i] = album.ID
|
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
var media []*models.Media
|
2020-12-08 16:24:08 +01:00
|
|
|
|
2020-12-17 21:32:13 +01:00
|
|
|
query := r.Database.
|
2020-12-08 16:24:08 +01:00
|
|
|
Joins("Album").
|
2020-12-22 01:14:43 +01:00
|
|
|
Where("albums.id IN (?)", userAlbumIDs).
|
2020-12-17 21:32:13 +01:00
|
|
|
Where("media.id IN (?)", r.Database.Model(&models.MediaURL{}).Select("id").Where("media_url.media_id = media.id"))
|
2020-12-08 16:24:08 +01:00
|
|
|
|
2020-12-17 21:32:13 +01:00
|
|
|
query = filter.FormatSQL(query)
|
|
|
|
|
|
|
|
if err := query.Scan(&media).Error; err != nil {
|
2020-02-09 14:21:53 +01:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
return media, nil
|
2020-02-05 14:51:46 +01:00
|
|
|
}
|
|
|
|
|
2020-07-10 14:26:19 +02:00
|
|
|
func (r *queryResolver) Media(ctx context.Context, id int) (*models.Media, error) {
|
2020-02-05 14:51:46 +01:00
|
|
|
user := auth.UserFromContext(ctx)
|
|
|
|
if user == nil {
|
|
|
|
return nil, auth.ErrUnauthorized
|
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
var media models.Media
|
2020-12-08 16:24:08 +01:00
|
|
|
|
|
|
|
err := r.Database.
|
|
|
|
Joins("Album").
|
|
|
|
Where("media.id = ?", id).
|
|
|
|
Where("Album.owner_id = ?", user.ID).
|
|
|
|
Where("media.id IN (?)", r.Database.Model(&models.MediaURL{}).Select("media_id").Where("media_urls.media_id = media.id")).
|
|
|
|
First(&media).Error
|
2020-02-05 14:51:46 +01:00
|
|
|
|
|
|
|
if err != nil {
|
2020-07-10 18:52:18 +02:00
|
|
|
return nil, errors.Wrap(err, "could not get media by media_id and user_id from database")
|
2020-02-05 14:51:46 +01:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
return &media, nil
|
2020-02-05 14:51:46 +01:00
|
|
|
}
|
|
|
|
|
2020-09-27 20:52:09 +02:00
|
|
|
func (r *queryResolver) MediaList(ctx context.Context, ids []int) ([]*models.Media, error) {
|
|
|
|
user := auth.UserFromContext(ctx)
|
|
|
|
if user == nil {
|
|
|
|
return nil, auth.ErrUnauthorized
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(ids) == 0 {
|
|
|
|
return nil, errors.New("no ids provided")
|
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
var media []*models.Media
|
2020-12-08 16:24:08 +01:00
|
|
|
err := r.Database.
|
|
|
|
Select("media.*").
|
|
|
|
Joins("Album").
|
|
|
|
Where("media.id IN ?", ids).
|
|
|
|
Where("album.owner_id = ?", user.ID).
|
|
|
|
Scan(&media).Error
|
|
|
|
|
2020-09-27 20:52:09 +02:00
|
|
|
if err != nil {
|
2020-11-28 17:31:19 +01:00
|
|
|
return nil, errors.Wrap(err, "could not get media list by media_id and user_id from database")
|
2020-09-27 20:52:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return media, nil
|
|
|
|
}
|
|
|
|
|
2020-07-10 14:26:19 +02:00
|
|
|
type mediaResolver struct {
|
2020-02-05 14:51:46 +01:00
|
|
|
*Resolver
|
|
|
|
}
|
|
|
|
|
2020-07-10 14:26:19 +02:00
|
|
|
func (r *Resolver) Media() api.MediaResolver {
|
|
|
|
return &mediaResolver{r}
|
2020-02-05 14:51:46 +01:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
func (r *mediaResolver) Shares(ctx context.Context, media *models.Media) ([]*models.ShareToken, error) {
|
|
|
|
var shareTokens []*models.ShareToken
|
2020-11-28 21:49:33 +01:00
|
|
|
if err := r.Database.Where("media_id = ?", media.ID).Find(&shareTokens).Error; err != nil {
|
2020-11-28 17:31:19 +01:00
|
|
|
return nil, errors.Wrapf(err, "get shares for media (%s)", media.Path)
|
2020-02-11 15:36:12 +01:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
return shareTokens, nil
|
2020-02-10 12:05:58 +01:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
func (r *mediaResolver) Downloads(ctx context.Context, media *models.Media) ([]*models.MediaDownload, error) {
|
2020-02-21 22:42:39 +01:00
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
var mediaUrls []*models.MediaURL
|
|
|
|
if err := r.Database.Where("media_id = ?", media.ID).Find(&mediaUrls).Error; err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "get downloads for media (%s)", media.Path)
|
2020-02-21 22:42:39 +01:00
|
|
|
}
|
|
|
|
|
2020-07-10 14:26:19 +02:00
|
|
|
downloads := make([]*models.MediaDownload, 0)
|
2020-02-21 22:42:39 +01:00
|
|
|
|
2020-07-10 14:26:19 +02:00
|
|
|
for _, url := range mediaUrls {
|
2020-02-21 22:42:39 +01:00
|
|
|
|
|
|
|
var title string
|
|
|
|
switch {
|
2020-07-10 12:58:11 +02:00
|
|
|
case url.Purpose == models.MediaOriginal:
|
2020-02-21 22:42:39 +01:00
|
|
|
title = "Original"
|
|
|
|
case url.Purpose == models.PhotoThumbnail:
|
|
|
|
title = "Small"
|
|
|
|
case url.Purpose == models.PhotoHighRes:
|
|
|
|
title = "Large"
|
2020-07-11 16:42:27 +02:00
|
|
|
case url.Purpose == models.VideoThumbnail:
|
|
|
|
title = "Video thumbnail"
|
|
|
|
case url.Purpose == models.VideoWeb:
|
|
|
|
title = "Web optimized video"
|
2020-02-21 22:42:39 +01:00
|
|
|
}
|
|
|
|
|
2020-07-10 14:26:19 +02:00
|
|
|
downloads = append(downloads, &models.MediaDownload{
|
2020-08-11 22:34:42 +02:00
|
|
|
Title: title,
|
|
|
|
MediaURL: url,
|
2020-02-21 22:42:39 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-02-10 12:05:58 +01:00
|
|
|
return downloads, nil
|
2020-02-09 12:53:21 +01:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
func (r *mediaResolver) HighRes(ctx context.Context, media *models.Media) (*models.MediaURL, error) {
|
|
|
|
var url models.MediaURL
|
|
|
|
err := r.Database.
|
2020-11-30 21:29:49 +01:00
|
|
|
Where("media_id = ?", media.ID).
|
2020-11-28 17:31:19 +01:00
|
|
|
Where("purpose = ? OR (purpose = ? AND content_type IN ?)", models.PhotoHighRes, models.MediaOriginal, scanner.WebMimetypes).
|
|
|
|
First(&url).Error
|
2020-02-05 14:51:46 +01:00
|
|
|
|
2020-02-09 14:21:53 +01:00
|
|
|
if err != nil {
|
2020-11-28 17:31:19 +01:00
|
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
2020-07-11 14:21:10 +02:00
|
|
|
return nil, nil
|
|
|
|
} else {
|
2020-11-28 17:31:19 +01:00
|
|
|
return nil, errors.Wrapf(err, "could not query high-res (%s)", media.Path)
|
2020-07-11 14:21:10 +02:00
|
|
|
}
|
2020-02-09 14:21:53 +01:00
|
|
|
}
|
2020-02-05 14:51:46 +01:00
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
return &url, nil
|
2020-02-05 14:51:46 +01:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
func (r *mediaResolver) Thumbnail(ctx context.Context, media *models.Media) (*models.MediaURL, error) {
|
|
|
|
var url models.MediaURL
|
|
|
|
err := r.Database.
|
2020-11-30 21:29:49 +01:00
|
|
|
Where("media_id = ?", media.ID).
|
2020-11-28 17:31:19 +01:00
|
|
|
Where("purpose = ? OR purpose = ?", models.PhotoThumbnail, models.VideoThumbnail).
|
|
|
|
First(&url).Error
|
2020-02-05 14:51:46 +01:00
|
|
|
|
2020-02-09 14:21:53 +01:00
|
|
|
if err != nil {
|
2020-11-28 17:31:19 +01:00
|
|
|
return nil, errors.Wrapf(err, "could not query thumbnail (%s)", media.Path)
|
2020-02-09 14:21:53 +01:00
|
|
|
}
|
2020-02-05 14:51:46 +01:00
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
return &url, nil
|
2020-02-05 14:51:46 +01:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
func (r *mediaResolver) VideoWeb(ctx context.Context, media *models.Media) (*models.MediaURL, error) {
|
|
|
|
|
|
|
|
var url models.MediaURL
|
|
|
|
err := r.Database.
|
2020-11-30 21:29:49 +01:00
|
|
|
Where("media_id = ?", media.ID).
|
2020-11-28 17:31:19 +01:00
|
|
|
Where("purpose = ?", models.VideoWeb).
|
|
|
|
First(&url).Error
|
2020-07-11 15:57:58 +02:00
|
|
|
|
|
|
|
if err != nil {
|
2020-11-28 17:31:19 +01:00
|
|
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
2020-07-11 16:42:27 +02:00
|
|
|
return nil, nil
|
|
|
|
} else {
|
2020-11-28 17:31:19 +01:00
|
|
|
return nil, errors.Wrapf(err, "could not query video web-format url (%s)", media.Path)
|
2020-07-11 16:42:27 +02:00
|
|
|
}
|
2020-07-11 15:57:58 +02:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
return &url, nil
|
2020-07-11 15:57:58 +02:00
|
|
|
}
|
|
|
|
|
2020-12-20 17:10:00 +01:00
|
|
|
func (r *mediaResolver) Favorite(ctx context.Context, media *models.Media) (bool, error) {
|
|
|
|
user := auth.UserFromContext(ctx)
|
|
|
|
if user == nil {
|
|
|
|
return false, auth.ErrUnauthorized
|
|
|
|
}
|
|
|
|
|
|
|
|
userMediaData := models.UserMediaData{
|
|
|
|
UserID: user.ID,
|
|
|
|
MediaID: media.ID,
|
|
|
|
Favorite: false,
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := r.Database.FirstOrInit(&userMediaData).Error; err != nil {
|
|
|
|
return false, errors.Wrapf(err, "get user media data from database (user: %d, media: %d)", user.ID, media.ID)
|
|
|
|
}
|
|
|
|
|
|
|
|
return userMediaData.Favorite, nil
|
|
|
|
}
|
|
|
|
|
2020-07-10 14:26:19 +02:00
|
|
|
func (r *mutationResolver) FavoriteMedia(ctx context.Context, mediaID int, favorite bool) (*models.Media, error) {
|
2020-06-17 18:00:58 +02:00
|
|
|
|
|
|
|
user := auth.UserFromContext(ctx)
|
2020-12-20 17:10:00 +01:00
|
|
|
if user == nil {
|
|
|
|
return nil, auth.ErrUnauthorized
|
|
|
|
}
|
2020-06-17 18:00:58 +02:00
|
|
|
|
2020-12-20 17:10:00 +01:00
|
|
|
userMediaData := models.UserMediaData{
|
|
|
|
UserID: user.ID,
|
|
|
|
MediaID: mediaID,
|
|
|
|
Favorite: favorite,
|
2020-06-17 18:00:58 +02:00
|
|
|
}
|
|
|
|
|
2020-12-20 17:10:00 +01:00
|
|
|
if err := r.Database.Clauses(clause.OnConflict{UpdateAll: true}).Create(&userMediaData).Error; err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "update user favorite media in database")
|
|
|
|
}
|
2020-11-28 17:31:19 +01:00
|
|
|
|
2020-12-20 17:10:00 +01:00
|
|
|
var media models.Media
|
|
|
|
if err := r.Database.First(&media, mediaID).Error; err != nil {
|
|
|
|
return nil, errors.Wrap(err, "get media from database after favorite update")
|
2020-06-19 15:34:29 +02:00
|
|
|
}
|
|
|
|
|
2020-11-28 17:31:19 +01:00
|
|
|
return &media, nil
|
2020-06-17 18:00:58 +02:00
|
|
|
}
|