mirror of
				https://github.com/zoriya/Kyoo.git
				synced 2025-10-26 08:12:35 -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
 | |
| }
 |