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

226 lines
6.9 KiB
Go
Raw Normal View History

2020-07-10 12:58:11 +02:00
package scanner
import (
2020-07-11 13:13:31 +02:00
"context"
"fmt"
2020-07-10 12:58:11 +02:00
"log"
"os"
2020-07-11 13:13:31 +02:00
"path"
"strings"
"time"
2020-07-10 12:58:11 +02:00
2020-12-17 22:51:43 +01:00
"github.com/photoview/photoview/api/graphql/models"
"github.com/photoview/photoview/api/scanner/media_encoding"
"github.com/photoview/photoview/api/scanner/media_encoding/executable_worker"
"github.com/photoview/photoview/api/scanner/media_encoding/media_utils"
2020-12-17 22:51:43 +01:00
"github.com/photoview/photoview/api/utils"
"github.com/pkg/errors"
2020-07-11 13:13:31 +02:00
"gopkg.in/vansante/go-ffprobe.v2"
2020-11-24 11:46:49 +01:00
"gorm.io/gorm"
2020-07-10 12:58:11 +02:00
)
func processVideo(tx *gorm.DB, mediaData *media_encoding.EncodeMediaData, videoCachePath *string) (bool, error) {
video := mediaData.Media
2020-07-10 12:58:11 +02:00
didProcess := false
log.Printf("Processing video: %s", video.Path)
2020-11-30 21:29:49 +01:00
mediaURLFromDB := makePhotoURLChecker(tx, video.ID)
2020-07-10 12:58:11 +02:00
videoOriginalURL, err := mediaURLFromDB(models.MediaOriginal)
if err != nil {
return false, errors.Wrap(err, "error processing video original format")
}
2020-11-30 21:29:49 +01:00
videoWebURL, err := mediaURLFromDB(models.VideoWeb)
2020-07-10 12:58:11 +02:00
if err != nil {
return false, errors.Wrap(err, "error processing video web-format")
}
2020-11-30 21:29:49 +01:00
videoThumbnailURL, err := mediaURLFromDB(models.VideoThumbnail)
2020-07-11 13:39:11 +02:00
if err != nil {
return false, errors.Wrap(err, "error processing video thumbnail")
}
videoType, err := mediaData.ContentType()
if err != nil {
return false, errors.Wrap(err, "error getting video content type")
}
if videoOriginalURL == nil && videoType.IsWebCompatible() {
didProcess = true
origVideoPath := video.Path
videoMediaName := generateUniqueMediaName(video.Path)
webMetadata, err := readVideoStreamMetadata(origVideoPath)
if err != nil {
return false, errors.Wrapf(err, "failed to read metadata for original video (%s)", video.Title)
}
fileStats, err := os.Stat(origVideoPath)
if err != nil {
return false, errors.Wrap(err, "reading file stats of original video")
}
mediaURL := models.MediaURL{
MediaID: video.ID,
MediaName: videoMediaName,
Width: webMetadata.Width,
Height: webMetadata.Height,
Purpose: models.MediaOriginal,
ContentType: string(*videoType),
FileSize: fileStats.Size(),
}
if err := tx.Create(&mediaURL).Error; err != nil {
return false, errors.Wrapf(err, "failed to insert original video into database (%s)", video.Title)
}
}
if videoWebURL == nil && !videoType.IsWebCompatible() {
2020-07-12 14:31:50 +02:00
didProcess = true
2020-07-11 13:13:31 +02:00
web_video_name := fmt.Sprintf("web_video_%s_%s", path.Base(video.Path), utils.GenerateToken())
web_video_name = strings.ReplaceAll(web_video_name, ".", "_")
web_video_name = strings.ReplaceAll(web_video_name, " ", "_")
web_video_name = web_video_name + ".mp4"
webVideoPath := path.Join(*videoCachePath, web_video_name)
err = executable_worker.FfmpegCli.EncodeMp4(video.Path, webVideoPath)
2020-07-11 13:13:31 +02:00
if err != nil {
return false, errors.Wrapf(err, "could not encode mp4 video (%s)", video.Path)
}
2020-07-11 13:39:11 +02:00
webMetadata, err := readVideoStreamMetadata(webVideoPath)
2020-07-11 13:13:31 +02:00
if err != nil {
return false, errors.Wrapf(err, "failed to read metadata for encoded web-video (%s)", video.Title)
}
fileStats, err := os.Stat(webVideoPath)
if err != nil {
return false, errors.Wrap(err, "reading file stats of web-optimized video")
}
2020-11-25 23:06:47 +01:00
mediaURL := models.MediaURL{
MediaID: video.ID,
MediaName: web_video_name,
Width: webMetadata.Width,
Height: webMetadata.Height,
Purpose: models.VideoWeb,
ContentType: "video/mp4",
FileSize: fileStats.Size(),
}
if err := tx.Create(&mediaURL).Error; err != nil {
2020-07-11 13:13:31 +02:00
return false, errors.Wrapf(err, "failed to insert encoded web-video into database (%s)", video.Title)
}
2020-07-10 12:58:11 +02:00
}
probeData, err := mediaData.VideoMetadata()
if err != nil {
return false, err
}
2020-07-11 13:39:11 +02:00
if videoThumbnailURL == nil {
2020-07-12 14:31:50 +02:00
didProcess = true
2020-07-11 13:39:11 +02:00
video_thumb_name := fmt.Sprintf("video_thumb_%s_%s", path.Base(video.Path), utils.GenerateToken())
video_thumb_name = strings.ReplaceAll(video_thumb_name, ".", "_")
video_thumb_name = strings.ReplaceAll(video_thumb_name, " ", "_")
video_thumb_name = video_thumb_name + ".jpg"
thumbImagePath := path.Join(*videoCachePath, video_thumb_name)
err = executable_worker.FfmpegCli.EncodeVideoThumbnail(video.Path, thumbImagePath, probeData)
2020-07-11 13:39:11 +02:00
if err != nil {
return false, errors.Wrapf(err, "failed to generate thumbnail for video (%s)", video.Title)
}
thumbDimensions, err := media_utils.GetPhotoDimensions(thumbImagePath)
2020-07-11 13:39:11 +02:00
if err != nil {
return false, errors.Wrap(err, "get dimensions of video thumbnail image")
}
fileStats, err := os.Stat(thumbImagePath)
if err != nil {
return false, errors.Wrap(err, "reading file stats of video thumbnail")
}
2020-11-25 23:06:47 +01:00
thumbMediaURL := models.MediaURL{
MediaID: video.ID,
MediaName: video_thumb_name,
Width: thumbDimensions.Width,
Height: thumbDimensions.Height,
Purpose: models.VideoThumbnail,
ContentType: "image/jpeg",
FileSize: fileStats.Size(),
}
if err := tx.Create(&thumbMediaURL).Error; err != nil {
2020-07-11 13:39:11 +02:00
return false, errors.Wrapf(err, "failed to insert video thumbnail image into database (%s)", video.Title)
}
} else {
// Verify that video thumbnail still exists in cache
thumbImagePath := path.Join(*videoCachePath, videoThumbnailURL.MediaName)
if _, err := os.Stat(thumbImagePath); os.IsNotExist(err) {
fmt.Printf("Video thumbnail found in database but not in cache, re-encoding photo to cache: %s\n", videoThumbnailURL.MediaName)
didProcess = true
err = executable_worker.FfmpegCli.EncodeVideoThumbnail(video.Path, thumbImagePath, probeData)
if err != nil {
return false, errors.Wrapf(err, "failed to generate thumbnail for video (%s)", video.Title)
}
thumbDimensions, err := media_utils.GetPhotoDimensions(thumbImagePath)
if err != nil {
return false, errors.Wrap(err, "get dimensions of video thumbnail image")
}
fileStats, err := os.Stat(thumbImagePath)
if err != nil {
return false, errors.Wrap(err, "reading file stats of video thumbnail")
}
videoThumbnailURL.Width = thumbDimensions.Width
videoThumbnailURL.Height = thumbDimensions.Height
videoThumbnailURL.FileSize = fileStats.Size()
if err := tx.Save(videoThumbnailURL).Error; err != nil {
return false, errors.Wrap(err, "updating video thumbnail url in database after re-encoding")
}
}
2020-07-11 13:39:11 +02:00
}
2020-07-10 12:58:11 +02:00
return didProcess, nil
}
2020-07-11 13:13:31 +02:00
2020-07-12 14:17:49 +02:00
func readVideoMetadata(videoPath string) (*ffprobe.ProbeData, error) {
2020-07-11 13:13:31 +02:00
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn()
data, err := ffprobe.ProbeURL(ctx, videoPath)
if err != nil {
return nil, errors.Wrapf(err, "could not read video metadata (%s)", path.Base(videoPath))
}
2020-07-12 14:17:49 +02:00
return data, nil
}
func readVideoStreamMetadata(videoPath string) (*ffprobe.Stream, error) {
data, err := readVideoMetadata(videoPath)
if err != nil {
return nil, errors.Wrap(err, "read video stream metadata")
}
2020-07-11 13:13:31 +02:00
stream := data.FirstVideoStream()
if stream == nil {
return nil, errors.Wrapf(err, "could not get stream from file metadata (%s)", path.Base(videoPath))
}
return stream, nil
}