mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 12:37:37 +03:00
161 lines
4.7 KiB
Go
161 lines
4.7 KiB
Go
package subsonic
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/navidrome/navidrome/conf"
|
|
"github.com/navidrome/navidrome/core"
|
|
"github.com/navidrome/navidrome/log"
|
|
"github.com/navidrome/navidrome/model"
|
|
"github.com/navidrome/navidrome/model/request"
|
|
"github.com/navidrome/navidrome/server/subsonic/responses"
|
|
"github.com/navidrome/navidrome/utils"
|
|
)
|
|
|
|
func (api *Router) serveStream(ctx context.Context, w http.ResponseWriter, r *http.Request, stream *core.Stream, id string) {
|
|
if stream.Seekable() {
|
|
http.ServeContent(w, r, stream.Name(), stream.ModTime(), stream)
|
|
} else {
|
|
// If the stream doesn't provide a size (i.e. is not seekable), we can't support ranges/content-length
|
|
w.Header().Set("Accept-Ranges", "none")
|
|
w.Header().Set("Content-Type", stream.ContentType())
|
|
|
|
estimateContentLength := utils.ParamBool(r, "estimateContentLength", false)
|
|
|
|
// if Client requests the estimated content-length, send it
|
|
if estimateContentLength {
|
|
length := strconv.Itoa(stream.EstimatedContentLength())
|
|
log.Trace(ctx, "Estimated content-length", "contentLength", length)
|
|
w.Header().Set("Content-Length", length)
|
|
}
|
|
|
|
if r.Method == http.MethodHead {
|
|
go func() { _, _ = io.Copy(io.Discard, stream) }()
|
|
} else {
|
|
c, err := io.Copy(w, stream)
|
|
if log.CurrentLevel() >= log.LevelDebug {
|
|
if err != nil {
|
|
log.Error(ctx, "Error sending transcoded file", "id", id, err)
|
|
} else {
|
|
log.Trace(ctx, "Success sending transcode file", "id", id, "size", c)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (api *Router) Stream(w http.ResponseWriter, r *http.Request) (*responses.Subsonic, error) {
|
|
ctx := r.Context()
|
|
id, err := requiredParamString(r, "id")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
maxBitRate := utils.ParamInt(r, "maxBitRate", 0)
|
|
format := utils.ParamString(r, "format")
|
|
timeOffset := utils.ParamInt(r, "timeOffset", 0)
|
|
|
|
stream, err := api.streamer.NewStream(ctx, id, format, maxBitRate, timeOffset)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Make sure the stream will be closed at the end, to avoid leakage
|
|
defer func() {
|
|
if err := stream.Close(); err != nil && log.CurrentLevel() >= log.LevelDebug {
|
|
log.Error("Error closing stream", "id", id, "file", stream.Name(), err)
|
|
}
|
|
}()
|
|
|
|
w.Header().Set("X-Content-Type-Options", "nosniff")
|
|
w.Header().Set("X-Content-Duration", strconv.FormatFloat(float64(stream.Duration()), 'G', -1, 32))
|
|
|
|
api.serveStream(ctx, w, r, stream, id)
|
|
|
|
return nil, nil
|
|
}
|
|
|
|
func (api *Router) Download(w http.ResponseWriter, r *http.Request) (*responses.Subsonic, error) {
|
|
ctx := r.Context()
|
|
username, _ := request.UsernameFrom(ctx)
|
|
id, err := requiredParamString(r, "id")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !conf.Server.EnableDownloads {
|
|
log.Warn(ctx, "Downloads are disabled", "user", username, "id", id)
|
|
return nil, newError(responses.ErrorAuthorizationFail, "downloads are disabled")
|
|
}
|
|
|
|
entity, err := model.GetEntityByID(ctx, api.ds, id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
maxBitRate := utils.ParamInt(r, "bitrate", 0)
|
|
format := utils.ParamString(r, "format")
|
|
|
|
if format == "" {
|
|
if conf.Server.AutoTranscodeDownload {
|
|
// if we are not provided a format, see if we have requested transcoding for this client
|
|
// This must be enabled via a config option. For the UI, we are always given an option.
|
|
// This will impact other clients which do not use the UI
|
|
transcoding, ok := request.TranscodingFrom(ctx)
|
|
|
|
if !ok {
|
|
format = "raw"
|
|
} else {
|
|
format = transcoding.TargetFormat
|
|
maxBitRate = transcoding.DefaultBitRate
|
|
}
|
|
} else {
|
|
format = "raw"
|
|
}
|
|
}
|
|
|
|
setHeaders := func(name string) {
|
|
name = strings.ReplaceAll(name, ",", "_")
|
|
disposition := fmt.Sprintf("attachment; filename=\"%s.zip\"", name)
|
|
w.Header().Set("Content-Disposition", disposition)
|
|
w.Header().Set("Content-Type", "application/zip")
|
|
}
|
|
|
|
switch v := entity.(type) {
|
|
case *model.MediaFile:
|
|
stream, err := api.streamer.NewStream(ctx, id, format, maxBitRate, 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Make sure the stream will be closed at the end, to avoid leakage
|
|
defer func() {
|
|
if err := stream.Close(); err != nil && log.CurrentLevel() >= log.LevelDebug {
|
|
log.Error("Error closing stream", "id", id, "file", stream.Name(), err)
|
|
}
|
|
}()
|
|
|
|
disposition := fmt.Sprintf("attachment; filename=\"%s\"", stream.Name())
|
|
w.Header().Set("Content-Disposition", disposition)
|
|
|
|
api.serveStream(ctx, w, r, stream, id)
|
|
return nil, nil
|
|
case *model.Album:
|
|
setHeaders(v.Name)
|
|
err = api.archiver.ZipAlbum(ctx, id, format, maxBitRate, w)
|
|
case *model.Artist:
|
|
setHeaders(v.Name)
|
|
err = api.archiver.ZipArtist(ctx, id, format, maxBitRate, w)
|
|
case *model.Playlist:
|
|
setHeaders(v.Name)
|
|
err = api.archiver.ZipPlaylist(ctx, id, format, maxBitRate, w)
|
|
default:
|
|
err = model.ErrNotFound
|
|
}
|
|
|
|
return nil, err
|
|
}
|