mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
76 lines
2 KiB
Go
76 lines
2 KiB
Go
package api
|
|
|
|
import (
|
|
"github.com/astaxie/beego"
|
|
"github.com/deluan/gosonic/api/responses"
|
|
"github.com/deluan/gosonic/domain"
|
|
"github.com/deluan/gosonic/engine"
|
|
"github.com/deluan/gosonic/utils"
|
|
)
|
|
|
|
type StreamController struct {
|
|
BaseAPIController
|
|
repo domain.MediaFileRepository
|
|
id string
|
|
mf *domain.MediaFile
|
|
}
|
|
|
|
func (c *StreamController) Prepare() {
|
|
utils.ResolveDependencies(&c.repo)
|
|
|
|
c.id = c.RequiredParamString("id", "id parameter required")
|
|
|
|
mf, err := c.repo.Get(c.id)
|
|
if err != nil {
|
|
beego.Error("Error reading mediafile", c.id, "from the database", ":", err)
|
|
c.SendError(responses.ERROR_GENERIC, "Internal error")
|
|
}
|
|
|
|
if mf == nil {
|
|
beego.Error("MediaFile", c.id, "not found!")
|
|
c.SendError(responses.ERROR_DATA_NOT_FOUND)
|
|
}
|
|
|
|
c.mf = mf
|
|
}
|
|
|
|
// TODO Still getting the "Conn.Write wrote more than the declared Content-Length" error.
|
|
// Don't know if this causes any issues
|
|
func (c *StreamController) Stream() {
|
|
maxBitRate := c.ParamInt("maxBitRate", 0)
|
|
maxBitRate = utils.MinInt(c.mf.BitRate, maxBitRate)
|
|
|
|
beego.Debug("Streaming file", c.id, ":", c.mf.Path)
|
|
beego.Debug("Bitrate", c.mf.BitRate, "MaxBitRate", maxBitRate)
|
|
|
|
// TODO Send proper estimated content-length
|
|
//contentLength := c.mf.Size
|
|
//if maxBitRate > 0 {
|
|
// contentLength = strconv.Itoa((c.mf.Duration + 1) * maxBitRate * 1000 / 8)
|
|
//}
|
|
c.Ctx.Output.Header("Content-Length", c.mf.Size)
|
|
c.Ctx.Output.Header("Content-Type", "audio/mpeg")
|
|
c.Ctx.Output.Header("Expires", "0")
|
|
c.Ctx.Output.Header("Cache-Control", "must-revalidate")
|
|
c.Ctx.Output.Header("Pragma", "public")
|
|
|
|
if c.Ctx.Request.Method == "HEAD" {
|
|
beego.Debug("Just a HEAD. Not streaming", c.mf.Path)
|
|
return
|
|
}
|
|
|
|
err := engine.Stream(c.mf.Path, c.mf.BitRate, maxBitRate, c.Ctx.ResponseWriter)
|
|
if err != nil {
|
|
beego.Error("Error streaming file", c.id, ":", err)
|
|
}
|
|
|
|
beego.Debug("Finished streaming of", c.mf.Path)
|
|
}
|
|
|
|
func (c *StreamController) Download() {
|
|
beego.Debug("Sending file", c.mf.Path)
|
|
|
|
engine.Stream(c.mf.Path, 0, 0, c.Ctx.ResponseWriter)
|
|
|
|
beego.Debug("Finished sending", c.mf.Path)
|
|
}
|