2019-02-10 05:30:54 +00:00
|
|
|
package manager
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"fmt"
|
|
|
|
"github.com/stashapp/stash/ffmpeg"
|
|
|
|
"github.com/stashapp/stash/logger"
|
|
|
|
"github.com/stashapp/stash/utils"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
)
|
|
|
|
|
|
|
|
type PreviewGenerator struct {
|
2019-02-10 09:44:12 +00:00
|
|
|
Info *GeneratorInfo
|
2019-02-10 05:30:54 +00:00
|
|
|
|
|
|
|
VideoFilename string
|
|
|
|
ImageFilename string
|
|
|
|
OutputDirectory string
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewPreviewGenerator(videoFile ffmpeg.VideoFile, videoFilename string, imageFilename string, outputDirectory string) (*PreviewGenerator, error) {
|
|
|
|
exists, err := utils.FileExists(videoFile.Path)
|
|
|
|
if !exists {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-02-10 09:44:12 +00:00
|
|
|
generator, err := newGeneratorInfo(videoFile)
|
2019-02-10 05:30:54 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
generator.ChunkCount = 12 // 12 segments to the preview
|
|
|
|
if err := generator.configure(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &PreviewGenerator{
|
2019-02-10 09:44:12 +00:00
|
|
|
Info: generator,
|
|
|
|
VideoFilename: videoFilename,
|
|
|
|
ImageFilename: imageFilename,
|
2019-02-10 05:30:54 +00:00
|
|
|
OutputDirectory: outputDirectory,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *PreviewGenerator) Generate() error {
|
|
|
|
instance.Paths.Generated.EmptyTmpDir()
|
2019-02-10 09:44:12 +00:00
|
|
|
logger.Infof("[generator] generating scene preview for %s", g.Info.VideoFile.Path)
|
2019-02-10 05:30:54 +00:00
|
|
|
encoder := ffmpeg.NewEncoder(instance.Paths.FixedPaths.FFMPEG)
|
|
|
|
|
|
|
|
if err := g.generateConcatFile(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := g.generateVideo(&encoder); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := g.generateImage(&encoder); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *PreviewGenerator) generateConcatFile() error {
|
|
|
|
f, err := os.Create(g.getConcatFilePath())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
w := bufio.NewWriter(f)
|
2019-02-10 09:44:12 +00:00
|
|
|
for i := 0; i < g.Info.ChunkCount; i++ {
|
2019-02-10 05:30:54 +00:00
|
|
|
num := fmt.Sprintf("%.3d", i)
|
|
|
|
filename := "preview"+num+".mp4"
|
|
|
|
_, _ = w.WriteString(fmt.Sprintf("file '%s'\n", filename))
|
|
|
|
}
|
|
|
|
return w.Flush()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *PreviewGenerator) generateVideo(encoder *ffmpeg.Encoder) error {
|
|
|
|
outputPath := path.Join(g.OutputDirectory, g.VideoFilename)
|
|
|
|
outputExists, _ := utils.FileExists(outputPath)
|
|
|
|
if outputExists {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-02-10 09:44:12 +00:00
|
|
|
stepSize := int(g.Info.VideoFile.Duration / float64(g.Info.ChunkCount))
|
|
|
|
for i := 0; i < g.Info.ChunkCount; i++ {
|
2019-02-10 05:30:54 +00:00
|
|
|
time := i * stepSize
|
|
|
|
num := fmt.Sprintf("%.3d", i)
|
|
|
|
filename := "preview"+num+".mp4"
|
|
|
|
chunkOutputPath := instance.Paths.Generated.GetTmpPath(filename)
|
|
|
|
|
|
|
|
options := ffmpeg.ScenePreviewChunkOptions{
|
|
|
|
Time: time,
|
|
|
|
Width: 640,
|
|
|
|
OutputPath: chunkOutputPath,
|
|
|
|
}
|
2019-02-10 09:44:12 +00:00
|
|
|
encoder.ScenePreviewVideoChunk(g.Info.VideoFile, options)
|
2019-02-10 05:30:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
videoOutputPath := path.Join(g.OutputDirectory, g.VideoFilename)
|
2019-02-10 09:44:12 +00:00
|
|
|
encoder.ScenePreviewVideoChunkCombine(g.Info.VideoFile, g.getConcatFilePath(), videoOutputPath)
|
2019-02-10 05:30:54 +00:00
|
|
|
logger.Debug("created video preview: ", videoOutputPath)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *PreviewGenerator) generateImage(encoder *ffmpeg.Encoder) error {
|
|
|
|
outputPath := path.Join(g.OutputDirectory, g.ImageFilename)
|
|
|
|
outputExists, _ := utils.FileExists(outputPath)
|
|
|
|
if outputExists {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
videoPreviewPath := path.Join(g.OutputDirectory, g.VideoFilename)
|
|
|
|
tmpOutputPath := instance.Paths.Generated.GetTmpPath(g.ImageFilename)
|
2019-02-10 09:44:12 +00:00
|
|
|
if err := encoder.ScenePreviewVideoToImage(g.Info.VideoFile, 640, videoPreviewPath, tmpOutputPath); err != nil {
|
2019-02-10 05:30:54 +00:00
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
if err := os.Rename(tmpOutputPath, outputPath); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
logger.Debug("created video preview image: ", outputPath)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (g *PreviewGenerator) getConcatFilePath() string {
|
|
|
|
return instance.Paths.Generated.GetTmpPath("files.txt")
|
|
|
|
}
|