Kyoo/transcoder/src/thumbnails.go
2024-12-24 22:55:42 +00:00

154 lines
3.5 KiB
Go

package src
import (
"encoding/base64"
"fmt"
"image"
"image/color"
"log"
"math"
"os"
"path/filepath"
"sync"
"github.com/disintegration/imaging"
"gitlab.com/opennota/screengen"
)
// We want to have a thumbnail every ${interval} seconds.
var default_interval = 10
// The maximim number of thumbnails per video.
// Setting this too high allows really long processing times.
var max_numcaps = 150
type Thumbnail struct {
ready sync.WaitGroup
path string
}
const ThumbsVersion = 1
func getThumbGlob(sha string) string {
return fmt.Sprintf("%s/%s/thumbs-v*.*", Settings.Metadata, sha)
}
func getThumbPath(sha string) string {
return fmt.Sprintf("%s/%s/thumbs-v%d.png", Settings.Metadata, sha, ThumbsVersion)
}
func getThumbVttPath(sha string) string {
return fmt.Sprintf("%s/%s/thumbs-v%d.vtt", Settings.Metadata, sha, ThumbsVersion)
}
func (s *MetadataService) GetThumb(path string, sha string) (string, string, error) {
_, err := s.ExtractThumbs(path, sha)
if err != nil {
return "", "", err
}
return getThumbPath(sha), getThumbVttPath(sha), nil
}
func (s *MetadataService) ExtractThumbs(path string, sha string) (interface{}, error) {
get_running, set := s.thumbLock.Start(sha)
if get_running != nil {
return get_running()
}
err := extractThumbnail(path, sha)
if err != nil {
return set(nil, err)
}
_, err = s.database.Exec(`update info set ver_thumbs = $2 where sha = $1`, sha, ThumbsVersion)
return set(nil, err)
}
func extractThumbnail(path string, sha string) error {
defer printExecTime("extracting thumbnails for %s", path)()
os.MkdirAll(fmt.Sprintf("%s/%s", Settings.Metadata, sha), 0o755)
if _, err := os.Stat(getThumbPath(sha)); err == nil {
return nil
}
gen, err := screengen.NewGenerator(path)
if err != nil {
log.Printf("Error reading video file: %v", err)
return err
}
defer gen.Close()
gen.Fast = true
duration := int(gen.Duration) / 1000
var numcaps int
if default_interval < duration {
numcaps = duration / default_interval
} else {
numcaps = duration / 10
}
numcaps = min(numcaps, max_numcaps)
interval := duration / numcaps
columns := int(math.Sqrt(float64(numcaps)))
rows := int(math.Ceil(float64(numcaps) / float64(columns)))
height := 144
width := int(float64(height) / float64(gen.Height()) * float64(gen.Width()))
sprite := imaging.New(width*columns, height*rows, color.Black)
vtt := "WEBVTT\n\n"
log.Printf("Extracting %d thumbnails for %s (interval of %d).", numcaps, path, interval)
ts := 0
for i := 0; i < numcaps; i++ {
img, err := gen.ImageWxH(int64(ts*1000), width, height)
if err != nil {
log.Printf("Could not generate screenshot %s", err)
return err
}
x := (i % columns) * width
y := (i / columns) * height
sprite = imaging.Paste(sprite, img, image.Pt(x, y))
timestamps := ts
ts += interval
vtt += fmt.Sprintf(
"%s --> %s\n%s/%s/thumbnails.png#xywh=%d,%d,%d,%d\n\n",
tsToVttTime(timestamps),
tsToVttTime(ts),
Settings.RoutePrefix,
base64.RawURLEncoding.EncodeToString([]byte(path)),
x,
y,
width,
height,
)
}
// Cleanup old versions of thumbnails
files, err := filepath.Glob(getThumbGlob(sha))
if err == nil {
for _, f := range files {
// ignore errors
os.Remove(f)
}
}
err = os.WriteFile(getThumbVttPath(sha), []byte(vtt), 0o644)
if err != nil {
return err
}
err = imaging.Save(sprite, getThumbPath(sha))
if err != nil {
return err
}
return nil
}
func tsToVttTime(ts int) string {
return fmt.Sprintf("%02d:%02d:%02d.000", ts/3600, (ts/60)%60, ts%60)
}