2023-05-23 18:44:05 -04:00
|
|
|
package ytdl
|
|
|
|
|
|
|
|
import (
|
2023-05-23 22:29:56 -04:00
|
|
|
"bytes"
|
2023-05-23 18:44:05 -04:00
|
|
|
"io"
|
2023-05-23 22:29:56 -04:00
|
|
|
"os/exec"
|
|
|
|
"strings"
|
2023-05-23 18:44:05 -04:00
|
|
|
|
|
|
|
"go.fifitido.net/ytdl-web/config"
|
2023-08-14 18:16:42 -04:00
|
|
|
"go.fifitido.net/ytdl-web/pkg/ytdl/cache"
|
|
|
|
"go.fifitido.net/ytdl-web/pkg/ytdl/metadata"
|
2023-05-23 18:44:05 -04:00
|
|
|
"golang.org/x/exp/slog"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Ytdl interface {
|
|
|
|
GetMetadata(url string) (*metadata.Metadata, error)
|
2023-05-23 19:35:52 -04:00
|
|
|
Download(w io.Writer, url, format string, index int) error
|
2023-05-23 22:29:56 -04:00
|
|
|
Version() string
|
2023-05-23 18:44:05 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
type ytdlImpl struct {
|
2023-05-23 22:29:56 -04:00
|
|
|
cfg *config.Config
|
|
|
|
logger *slog.Logger
|
|
|
|
cache cache.MetadataCache
|
|
|
|
version string
|
2023-05-23 18:44:05 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewYtdl(cfg *config.Config, logger *slog.Logger, cache cache.MetadataCache) Ytdl {
|
2023-05-23 22:29:56 -04:00
|
|
|
cmd := exec.Command(
|
|
|
|
cfg.BinaryPath,
|
|
|
|
"--version",
|
|
|
|
)
|
|
|
|
var out bytes.Buffer
|
|
|
|
cmd.Stdout = &out
|
|
|
|
_ = cmd.Run()
|
|
|
|
|
2023-05-23 18:44:05 -04:00
|
|
|
return &ytdlImpl{
|
2023-05-23 22:29:56 -04:00
|
|
|
cfg: cfg,
|
|
|
|
logger: logger.With(slog.String("module", "ytdl")),
|
|
|
|
cache: cache,
|
|
|
|
version: strings.TrimSpace(out.String()),
|
2023-05-23 18:44:05 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (y *ytdlImpl) baseOptions(url string) []Option {
|
|
|
|
options := []Option{}
|
|
|
|
|
|
|
|
metadata, err := y.cache.Get(url)
|
|
|
|
if err == nil {
|
|
|
|
options = append(options, WithLoadJson(metadata))
|
|
|
|
}
|
|
|
|
|
|
|
|
if y.cfg.Cookies.Enabled {
|
2023-05-23 21:52:00 -04:00
|
|
|
if y.cfg.Cookies.FromBrowser.Browser != "" {
|
2023-05-23 18:44:05 -04:00
|
|
|
options = append(options, WithBrowserCookies(
|
|
|
|
y.cfg.Cookies.FromBrowser.Browser,
|
|
|
|
y.cfg.Cookies.FromBrowser.Keyring,
|
|
|
|
y.cfg.Cookies.FromBrowser.Profile,
|
|
|
|
y.cfg.Cookies.FromBrowser.Container,
|
|
|
|
))
|
|
|
|
} else {
|
|
|
|
options = append(options, WithCookieFile(y.cfg.Cookies.FilePath))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-23 19:07:44 -04:00
|
|
|
if y.cfg.IsDevelopment() {
|
|
|
|
options = append(options, WithDebug())
|
|
|
|
}
|
|
|
|
|
2023-05-23 18:44:05 -04:00
|
|
|
return options
|
|
|
|
}
|
|
|
|
|
2023-05-23 22:29:56 -04:00
|
|
|
func (y *ytdlImpl) Version() string {
|
|
|
|
return y.version
|
|
|
|
}
|
|
|
|
|
2023-05-23 18:44:05 -04:00
|
|
|
// GetMetadata implements Ytdl
|
|
|
|
func (y *ytdlImpl) GetMetadata(url string) (*metadata.Metadata, error) {
|
2023-05-23 19:13:48 -04:00
|
|
|
meta, err := y.cache.Get(url)
|
|
|
|
if err == nil {
|
|
|
|
return meta, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
meta = &metadata.Metadata{}
|
2023-05-23 18:44:05 -04:00
|
|
|
options := append(
|
|
|
|
y.baseOptions(url),
|
|
|
|
WithDumpJson(meta),
|
|
|
|
)
|
|
|
|
|
2023-05-23 20:07:48 -04:00
|
|
|
if err := Exec(y.cfg.BinaryPath, url, options...); err != nil {
|
2023-05-23 19:13:48 -04:00
|
|
|
y.logger.Error("failed to get metadata", slog.String("url", url), slog.String("error", err.Error()))
|
2023-05-23 18:44:05 -04:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-05-23 20:07:48 -04:00
|
|
|
if err := y.cache.Set(url, meta, y.cfg.Cache.TTL); err != nil {
|
2023-05-23 18:44:05 -04:00
|
|
|
y.logger.Warn("failed to cache metadata", slog.String("url", url), slog.String("error", err.Error()))
|
|
|
|
}
|
|
|
|
|
|
|
|
return meta, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Download implements Ytdl
|
2023-05-23 19:35:52 -04:00
|
|
|
func (y *ytdlImpl) Download(w io.Writer, url, format string, index int) error {
|
2023-05-23 18:44:05 -04:00
|
|
|
options := append(
|
|
|
|
y.baseOptions(url),
|
|
|
|
WithFormat(format),
|
|
|
|
WithStreamOutput(w),
|
|
|
|
)
|
|
|
|
|
2023-05-23 19:35:52 -04:00
|
|
|
if index >= 0 {
|
|
|
|
options = append(options, WithPlaylistIndex(index))
|
|
|
|
}
|
|
|
|
|
2023-05-23 20:07:48 -04:00
|
|
|
if err := Exec(y.cfg.BinaryPath, url, options...); err != nil {
|
2023-05-23 18:44:05 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|