2021-02-18 23:05:52 -08:00
|
|
|
package transcoder
|
2020-06-10 01:16:17 -07:00
|
|
|
|
|
|
|
import (
|
2021-07-28 23:21:02 +02:00
|
|
|
"os"
|
2020-06-10 01:16:17 -07:00
|
|
|
"os/exec"
|
|
|
|
"path"
|
2020-07-06 19:45:58 -07:00
|
|
|
"strconv"
|
2020-06-18 17:47:44 -07:00
|
|
|
"strings"
|
2020-06-10 01:16:17 -07:00
|
|
|
"time"
|
2020-06-22 13:41:53 -07:00
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
2020-06-22 20:11:56 -05:00
|
|
|
|
2020-10-06 01:07:09 +08:00
|
|
|
"github.com/owncast/owncast/config"
|
2021-02-18 23:05:52 -08:00
|
|
|
"github.com/owncast/owncast/core/data"
|
|
|
|
"github.com/owncast/owncast/utils"
|
2020-06-10 01:16:17 -07:00
|
|
|
)
|
|
|
|
|
2020-10-14 14:07:38 -07:00
|
|
|
var _timer *time.Ticker
|
|
|
|
|
2021-09-12 00:18:15 -07:00
|
|
|
// StopThumbnailGenerator will stop the periodic generating of a thumbnail from video.
|
2020-10-14 14:07:38 -07:00
|
|
|
func StopThumbnailGenerator() {
|
|
|
|
if _timer != nil {
|
|
|
|
_timer.Stop()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// StartThumbnailGenerator starts generating thumbnails.
|
2020-07-06 19:45:58 -07:00
|
|
|
func StartThumbnailGenerator(chunkPath string, variantIndex int) {
|
2020-06-10 01:16:17 -07:00
|
|
|
// Every 20 seconds create a thumbnail from the most
|
|
|
|
// recent video segment.
|
2020-10-14 14:07:38 -07:00
|
|
|
_timer = time.NewTicker(20 * time.Second)
|
2020-06-10 01:16:17 -07:00
|
|
|
quit := make(chan struct{})
|
2020-06-22 20:11:56 -05:00
|
|
|
|
2020-06-10 01:16:17 -07:00
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
2020-10-14 14:07:38 -07:00
|
|
|
case <-_timer.C:
|
2020-07-06 19:45:58 -07:00
|
|
|
if err := fireThumbnailGenerator(chunkPath, variantIndex); err != nil {
|
2020-06-22 20:11:56 -05:00
|
|
|
log.Errorln("Unable to generate thumbnail:", err)
|
|
|
|
}
|
2020-06-10 01:16:17 -07:00
|
|
|
case <-quit:
|
2020-07-06 21:27:31 -07:00
|
|
|
log.Debug("thumbnail generator has stopped")
|
2020-10-14 14:07:38 -07:00
|
|
|
_timer.Stop()
|
2020-06-10 01:16:17 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-10-14 14:07:38 -07:00
|
|
|
func fireThumbnailGenerator(segmentPath string, variantIndex int) error {
|
2020-06-18 17:47:44 -07:00
|
|
|
// JPG takes less time to encode than PNG
|
2020-10-03 14:35:03 -07:00
|
|
|
outputFile := path.Join(config.WebRoot, "thumbnail.jpg")
|
|
|
|
previewGifFile := path.Join(config.WebRoot, "preview.gif")
|
2020-06-10 01:16:17 -07:00
|
|
|
|
2020-10-14 14:07:38 -07:00
|
|
|
framePath := path.Join(segmentPath, strconv.Itoa(variantIndex))
|
2021-11-20 14:42:50 +08:00
|
|
|
files, err := os.ReadDir(framePath)
|
2020-06-10 01:16:17 -07:00
|
|
|
if err != nil {
|
2020-06-22 20:11:56 -05:00
|
|
|
return err
|
2020-06-10 01:16:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
var modTime time.Time
|
|
|
|
var names []string
|
2021-11-20 14:42:50 +08:00
|
|
|
for _, f := range files {
|
|
|
|
if path.Ext(f.Name()) != ".ts" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
fi, err := f.Info()
|
|
|
|
if err != nil {
|
2020-06-10 01:16:17 -07:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if fi.Mode().IsRegular() {
|
|
|
|
if !fi.ModTime().Before(modTime) {
|
|
|
|
if fi.ModTime().After(modTime) {
|
|
|
|
modTime = fi.ModTime()
|
|
|
|
names = names[:0]
|
|
|
|
}
|
|
|
|
names = append(names, fi.Name())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(names) == 0 {
|
2020-06-22 20:11:56 -05:00
|
|
|
return nil
|
2020-06-10 01:16:17 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
mostRecentFile := path.Join(framePath, names[0])
|
2021-02-18 23:05:52 -08:00
|
|
|
ffmpegPath := utils.ValidatedFfmpegPath(data.GetFfMpegPath())
|
2021-07-28 23:21:02 +02:00
|
|
|
outputFileTemp := path.Join(config.WebRoot, "tempthumbnail.jpg")
|
2020-06-10 01:16:17 -07:00
|
|
|
|
2020-06-18 17:47:44 -07:00
|
|
|
thumbnailCmdFlags := []string{
|
2021-02-18 23:05:52 -08:00
|
|
|
ffmpegPath,
|
2020-06-18 17:47:44 -07:00
|
|
|
"-y", // Overwrite file
|
|
|
|
"-threads 1", // Low priority processing
|
2020-07-06 22:39:07 -07:00
|
|
|
"-t 1", // Pull from frame 1
|
2020-06-18 17:47:44 -07:00
|
|
|
"-i", mostRecentFile, // Input
|
|
|
|
"-f image2", // format
|
|
|
|
"-vframes 1", // Single frame
|
2021-07-28 23:21:02 +02:00
|
|
|
outputFileTemp,
|
2020-06-18 17:47:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ffmpegCmd := strings.Join(thumbnailCmdFlags, " ")
|
2020-06-22 20:11:56 -05:00
|
|
|
if _, err := exec.Command("sh", "-c", ffmpegCmd).Output(); err != nil {
|
|
|
|
return err
|
2021-09-12 00:18:15 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// rename temp file
|
|
|
|
if err := os.Rename(outputFileTemp, outputFile); err != nil {
|
|
|
|
log.Errorln(err)
|
2020-06-22 13:41:53 -07:00
|
|
|
}
|
2020-06-22 20:11:56 -05:00
|
|
|
|
2020-10-01 23:55:38 -07:00
|
|
|
// If YP support is enabled also create an animated GIF preview
|
2021-02-18 23:05:52 -08:00
|
|
|
if data.GetDirectoryEnabled() {
|
2020-10-01 23:55:38 -07:00
|
|
|
makeAnimatedGifPreview(mostRecentFile, previewGifFile)
|
|
|
|
}
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
return nil
|
2020-06-10 01:16:17 -07:00
|
|
|
}
|
2020-10-01 23:55:38 -07:00
|
|
|
|
|
|
|
func makeAnimatedGifPreview(sourceFile string, outputFile string) {
|
2021-02-18 23:05:52 -08:00
|
|
|
ffmpegPath := utils.ValidatedFfmpegPath(data.GetFfMpegPath())
|
2021-07-28 23:21:02 +02:00
|
|
|
outputFileTemp := path.Join(config.WebRoot, "temppreview.gif")
|
2021-02-18 23:05:52 -08:00
|
|
|
|
2020-10-01 23:55:38 -07:00
|
|
|
// Filter is pulled from https://engineering.giphy.com/how-to-make-gifs-with-ffmpeg/
|
|
|
|
animatedGifFlags := []string{
|
2021-02-18 23:05:52 -08:00
|
|
|
ffmpegPath,
|
2020-10-01 23:55:38 -07:00
|
|
|
"-y", // Overwrite file
|
|
|
|
"-threads 1", // Low priority processing
|
|
|
|
"-i", sourceFile, // Input
|
|
|
|
"-t 1", // Output is one second in length
|
|
|
|
"-filter_complex", "\"[0:v] fps=8,scale=w=480:h=-1:flags=lanczos,split [a][b];[a] palettegen=stats_mode=full [p];[b][p] paletteuse=new=1\"",
|
2021-07-28 23:21:02 +02:00
|
|
|
outputFileTemp,
|
2020-10-01 23:55:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ffmpegCmd := strings.Join(animatedGifFlags, " ")
|
|
|
|
if _, err := exec.Command("sh", "-c", ffmpegCmd).Output(); err != nil {
|
|
|
|
log.Errorln(err)
|
2021-07-28 23:21:02 +02:00
|
|
|
// rename temp file
|
2021-10-25 00:31:45 -07:00
|
|
|
} else if err := os.Rename(outputFileTemp, outputFile); err != nil {
|
|
|
|
log.Errorln(err)
|
2020-10-01 23:55:38 -07:00
|
|
|
}
|
|
|
|
}
|