mirror of
https://github.com/zoriya/Kyoo.git
synced 2025-10-23 23:09:09 -04:00
407 lines
9.8 KiB
Go
407 lines
9.8 KiB
Go
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"mime"
|
|
"net/http"
|
|
"path/filepath"
|
|
"strconv"
|
|
|
|
"github.com/zoriya/kyoo/transcoder/src"
|
|
"github.com/zoriya/kyoo/transcoder/src/api"
|
|
"github.com/zoriya/kyoo/transcoder/src/utils"
|
|
|
|
"github.com/labstack/echo/v4"
|
|
"github.com/labstack/echo/v4/middleware"
|
|
)
|
|
|
|
// Direct video
|
|
//
|
|
// Retrieve the raw video stream, in the same container as the one on the server. No transcoding or
|
|
// transmuxing is done.
|
|
//
|
|
// Path: /:path/direct
|
|
func DirectStream(c echo.Context) error {
|
|
path, _, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.File(path)
|
|
}
|
|
|
|
// Get master playlist
|
|
//
|
|
// Get a master playlist containing all possible video qualities and audios available for this resource.
|
|
// Note that the direct stream is missing (since the direct is not an hls stream) and
|
|
// subtitles/fonts are not included to support more codecs than just webvtt.
|
|
//
|
|
// Path: /:path/master.m3u8
|
|
func (h *Handler) GetMaster(c echo.Context) error {
|
|
client, err := GetClientId(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
path, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ret, err := h.transcoder.GetMaster(c.Request().Context(), path, client, sha)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.String(http.StatusOK, ret)
|
|
}
|
|
|
|
// Transcode video
|
|
//
|
|
// Transcode the video to the selected quality.
|
|
// This route can take a few seconds to respond since it will way for at least one segment to be
|
|
// available.
|
|
//
|
|
// Path: /:path/:video/:quality/index.m3u8
|
|
func (h *Handler) GetVideoIndex(c echo.Context) error {
|
|
video, err := strconv.ParseInt(c.Param("video"), 10, 32)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
quality, err := src.QualityFromString(c.Param("quality"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
client, err := GetClientId(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
path, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ret, err := h.transcoder.GetVideoIndex(c.Request().Context(), path, uint32(video), quality, client, sha)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.String(http.StatusOK, ret)
|
|
}
|
|
|
|
// Transcode audio
|
|
//
|
|
// Get the selected audio
|
|
// This route can take a few seconds to respond since it will way for at least one segment to be
|
|
// available.
|
|
//
|
|
// Path: /:path/audio/:audio/index.m3u8
|
|
func (h *Handler) GetAudioIndex(c echo.Context) error {
|
|
audio, err := strconv.ParseInt(c.Param("audio"), 10, 32)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
client, err := GetClientId(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
path, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ret, err := h.transcoder.GetAudioIndex(c.Request().Context(), path, uint32(audio), client, sha)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.String(http.StatusOK, ret)
|
|
}
|
|
|
|
// Get transmuxed chunk
|
|
//
|
|
// Retrieve a chunk of a transmuxed video.
|
|
//
|
|
// Path: /:path/:video/:quality/segments-:chunk.ts
|
|
func (h *Handler) GetVideoSegment(c echo.Context) error {
|
|
video, err := strconv.ParseInt(c.Param("video"), 10, 32)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
quality, err := src.QualityFromString(c.Param("quality"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
segment, err := ParseSegment(c.Param("chunk"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
client, err := GetClientId(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
path, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ret, err := h.transcoder.GetVideoSegment(
|
|
c.Request().Context(),
|
|
path,
|
|
uint32(video),
|
|
quality,
|
|
segment,
|
|
client,
|
|
sha,
|
|
)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.File(ret)
|
|
}
|
|
|
|
// Get audio chunk
|
|
//
|
|
// Retrieve a chunk of a transcoded audio.
|
|
//
|
|
// Path: /:path/audio/:audio/segments-:chunk.ts
|
|
func (h *Handler) GetAudioSegment(c echo.Context) error {
|
|
audio, err := strconv.ParseInt(c.Param("audio"), 10, 32)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
segment, err := ParseSegment(c.Param("chunk"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
client, err := GetClientId(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
path, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ret, err := h.transcoder.GetAudioSegment(c.Request().Context(), path, uint32(audio), segment, client, sha)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.File(ret)
|
|
}
|
|
|
|
// Identify
|
|
//
|
|
// Identify metadata about a file.
|
|
//
|
|
// Path: /:path/info
|
|
func (h *Handler) GetInfo(c echo.Context) error {
|
|
path, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ret, err := h.metadata.GetMetadata(c.Request().Context(), path, sha)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = ret.SearchExternalSubtitles()
|
|
if err != nil {
|
|
fmt.Printf("Couldn't find external subtitles: %v", err)
|
|
}
|
|
return c.JSON(http.StatusOK, ret)
|
|
}
|
|
|
|
// Get attachments
|
|
//
|
|
// Get a specific attachment.
|
|
//
|
|
// Path: /:path/attachment/:name
|
|
func (h *Handler) GetAttachment(c echo.Context) (err error) {
|
|
_, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
name := c.Param("name")
|
|
if err := SanitizePath(name); err != nil {
|
|
return err
|
|
}
|
|
|
|
attachementStream, err := h.metadata.GetAttachment(c.Request().Context(), sha, name)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer utils.CleanupWithErr(&err, attachementStream.Close, "failed to close attachment reader")
|
|
|
|
mimeType, err := guessMimeType(name, attachementStream)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to guess mime type: %w", err)
|
|
}
|
|
|
|
return c.Stream(200, mimeType, attachementStream)
|
|
}
|
|
|
|
// Get subtitle
|
|
//
|
|
// Get a specific subtitle.
|
|
//
|
|
// Path: /:path/subtitle/:name
|
|
func (h *Handler) GetSubtitle(c echo.Context) (err error) {
|
|
_, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
name := c.Param("name")
|
|
if err := SanitizePath(name); err != nil {
|
|
return err
|
|
}
|
|
|
|
subtitleStream, err := h.metadata.GetSubtitle(c.Request().Context(), sha, name)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer utils.CleanupWithErr(&err, subtitleStream.Close, "failed to close subtitle reader")
|
|
|
|
mimeType, err := guessMimeType(name, subtitleStream)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to guess mime type: %w", err)
|
|
}
|
|
|
|
// Default the mime type to text/plain if it is not recognized
|
|
if mimeType == "" {
|
|
mimeType = "text/plain"
|
|
}
|
|
|
|
return c.Stream(200, mimeType, subtitleStream)
|
|
}
|
|
|
|
// Get thumbnail sprite
|
|
//
|
|
// Get a sprite file containing all the thumbnails of the show.
|
|
//
|
|
// Path: /:path/thumbnails.png
|
|
func (h *Handler) GetThumbnails(c echo.Context) (err error) {
|
|
path, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
sprite, err := h.metadata.GetThumbSprite(c.Request().Context(), path, sha)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer utils.CleanupWithErr(&err, sprite.Close, "failed to close thumbnail sprite reader")
|
|
|
|
return c.Stream(200, "image/png", sprite)
|
|
}
|
|
|
|
// Get thumbnail vtt
|
|
//
|
|
// Get a vtt file containing timing/position of thumbnails inside the sprite file.
|
|
// https://developer.bitmovin.com/playback/docs/webvtt-based-thumbnails for more info.
|
|
//
|
|
// Path: /:path/:resource/:slug/thumbnails.vtt
|
|
func (h *Handler) GetThumbnailsVtt(c echo.Context) (err error) {
|
|
path, sha, err := GetPath(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
vtt, err := h.metadata.GetThumbVtt(c.Request().Context(), path, sha)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer utils.CleanupWithErr(&err, vtt.Close, "failed to close thumbnail vtt reader")
|
|
|
|
return c.Stream(200, "text/vtt", vtt)
|
|
}
|
|
|
|
type Handler struct {
|
|
transcoder *src.Transcoder
|
|
metadata *src.MetadataService
|
|
}
|
|
|
|
// Try to guess the mime type of a file based on its extension.
|
|
// If the extension is not recognized, return an empty string.
|
|
// If path is provided, it should contain a file extension (i.e. ".mp4").
|
|
// If content is provided, it should be of type io.ReadSeeker. Instances of other types are ignored.
|
|
// This implementation is based upon http.ServeContent.
|
|
func guessMimeType(path string, content any) (string, error) {
|
|
// This does not match a large number of different types that are likely in use.
|
|
// TODO add telemetry to see what file extensions are used, then add logic
|
|
// to detect the type based on the file extension.
|
|
mimeType := ""
|
|
|
|
// First check the file extension, if there is one.
|
|
ext := filepath.Ext(path)
|
|
if ext != "" {
|
|
if mimeType = mime.TypeByExtension(ext); mimeType != "" {
|
|
return mimeType, nil
|
|
}
|
|
}
|
|
|
|
// Try reading the first few bytes of the file to guess the mime type.
|
|
// Only do this if seeking is supported
|
|
if reader, ok := content.(io.ReadSeeker); ok {
|
|
// 512 bytes is the most that DetectContentType will consider, so no
|
|
// need to read more than that.
|
|
var buf [512]byte
|
|
n, _ := io.ReadFull(reader, buf[:])
|
|
mimeType = http.DetectContentType(buf[:n])
|
|
|
|
// Reset the reader to the beginning of the file
|
|
_, err := reader.Seek(0, io.SeekStart)
|
|
if err != nil {
|
|
return "", fmt.Errorf("mime type guesser failed to seek to beginning of file: %w", err)
|
|
}
|
|
}
|
|
|
|
return mimeType, nil
|
|
}
|
|
|
|
func main() {
|
|
e := echo.New()
|
|
|
|
if err := run(e); err != nil {
|
|
e.Logger.Fatal(err)
|
|
}
|
|
}
|
|
|
|
func run(e *echo.Echo) (err error) {
|
|
e.Use(middleware.Logger())
|
|
e.HTTPErrorHandler = ErrorHandler
|
|
|
|
metadata, err := src.NewMetadataService()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create metadata service: %w", err)
|
|
}
|
|
defer utils.CleanupWithErr(&err, metadata.Close, "failed to close metadata service")
|
|
|
|
transcoder, err := src.NewTranscoder(metadata)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create transcoder: %w", err)
|
|
}
|
|
|
|
h := Handler{
|
|
transcoder: transcoder,
|
|
metadata: metadata,
|
|
}
|
|
|
|
g := e.Group(src.Settings.RoutePrefix)
|
|
g.GET("/:path/direct", DirectStream)
|
|
g.GET("/:path/direct/:identifier", DirectStream)
|
|
g.GET("/:path/master.m3u8", h.GetMaster)
|
|
g.GET("/:path/:video/:quality/index.m3u8", h.GetVideoIndex)
|
|
g.GET("/:path/audio/:audio/index.m3u8", h.GetAudioIndex)
|
|
g.GET("/:path/:video/:quality/:chunk", h.GetVideoSegment)
|
|
g.GET("/:path/audio/:audio/:chunk", h.GetAudioSegment)
|
|
g.GET("/:path/info", h.GetInfo)
|
|
g.GET("/:path/thumbnails.png", h.GetThumbnails)
|
|
g.GET("/:path/thumbnails.vtt", h.GetThumbnailsVtt)
|
|
g.GET("/:path/attachment/:name", h.GetAttachment)
|
|
g.GET("/:path/subtitle/:name", h.GetSubtitle)
|
|
|
|
api.RegisterPProfHandlers(e)
|
|
|
|
if err := e.Start(":7666"); err != nil {
|
|
return fmt.Errorf("failed to start server: %w", err)
|
|
}
|
|
return nil
|
|
}
|