mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
270 lines
9.1 KiB
Go
270 lines
9.1 KiB
Go
package subsonic
|
|
|
|
import (
|
|
"encoding/json"
|
|
"encoding/xml"
|
|
"fmt"
|
|
"net/http"
|
|
"runtime"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
"github.com/go-chi/chi/v5/middleware"
|
|
"github.com/navidrome/navidrome/consts"
|
|
"github.com/navidrome/navidrome/core"
|
|
"github.com/navidrome/navidrome/core/scrobbler"
|
|
"github.com/navidrome/navidrome/log"
|
|
"github.com/navidrome/navidrome/model"
|
|
"github.com/navidrome/navidrome/scanner"
|
|
"github.com/navidrome/navidrome/server/events"
|
|
"github.com/navidrome/navidrome/server/subsonic/responses"
|
|
"github.com/navidrome/navidrome/utils"
|
|
)
|
|
|
|
const Version = "1.16.1"
|
|
|
|
type handler = func(http.ResponseWriter, *http.Request) (*responses.Subsonic, error)
|
|
|
|
type Router struct {
|
|
http.Handler
|
|
DataStore model.DataStore
|
|
Artwork core.Artwork
|
|
Streamer core.MediaStreamer
|
|
Archiver core.Archiver
|
|
Players core.Players
|
|
ExternalMetadata core.ExternalMetadata
|
|
Playlists core.Playlists
|
|
Scanner scanner.Scanner
|
|
Broker events.Broker
|
|
Scrobbler scrobbler.PlayTracker
|
|
}
|
|
|
|
func New(ds model.DataStore, artwork core.Artwork, streamer core.MediaStreamer, archiver core.Archiver,
|
|
players core.Players, externalMetadata core.ExternalMetadata, scanner scanner.Scanner, broker events.Broker,
|
|
playlists core.Playlists, scrobbler scrobbler.PlayTracker) *Router {
|
|
r := &Router{
|
|
DataStore: ds,
|
|
Artwork: artwork,
|
|
Streamer: streamer,
|
|
Archiver: archiver,
|
|
Players: players,
|
|
ExternalMetadata: externalMetadata,
|
|
Playlists: playlists,
|
|
Scanner: scanner,
|
|
Broker: broker,
|
|
Scrobbler: scrobbler,
|
|
}
|
|
r.Handler = r.routes()
|
|
return r
|
|
}
|
|
|
|
func (api *Router) routes() http.Handler {
|
|
r := chi.NewRouter()
|
|
|
|
r.Use(postFormToQueryParams)
|
|
r.Use(checkRequiredParameters)
|
|
r.Use(authenticate(api.DataStore))
|
|
// TODO Validate version
|
|
|
|
// Subsonic endpoints, grouped by controller
|
|
r.Group(func(r chi.Router) {
|
|
c := initSystemController(api)
|
|
withPlayer := r.With(getPlayer(api.Players))
|
|
h(withPlayer, "ping", c.Ping)
|
|
h(withPlayer, "getLicense", c.GetLicense)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initBrowsingController(api)
|
|
withPlayer := r.With(getPlayer(api.Players))
|
|
h(withPlayer, "getMusicFolders", c.GetMusicFolders)
|
|
h(withPlayer, "getIndexes", c.GetIndexes)
|
|
h(withPlayer, "getArtists", c.GetArtists)
|
|
h(withPlayer, "getGenres", c.GetGenres)
|
|
h(withPlayer, "getMusicDirectory", c.GetMusicDirectory)
|
|
h(withPlayer, "getArtist", c.GetArtist)
|
|
h(withPlayer, "getAlbum", c.GetAlbum)
|
|
h(withPlayer, "getSong", c.GetSong)
|
|
h(withPlayer, "getArtistInfo", c.GetArtistInfo)
|
|
h(withPlayer, "getArtistInfo2", c.GetArtistInfo2)
|
|
h(withPlayer, "getTopSongs", c.GetTopSongs)
|
|
h(withPlayer, "getSimilarSongs", c.GetSimilarSongs)
|
|
h(withPlayer, "getSimilarSongs2", c.GetSimilarSongs2)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initAlbumListController(api)
|
|
withPlayer := r.With(getPlayer(api.Players))
|
|
h(withPlayer, "getAlbumList", c.GetAlbumList)
|
|
h(withPlayer, "getAlbumList2", c.GetAlbumList2)
|
|
h(withPlayer, "getStarred", c.GetStarred)
|
|
h(withPlayer, "getStarred2", c.GetStarred2)
|
|
h(withPlayer, "getNowPlaying", c.GetNowPlaying)
|
|
h(withPlayer, "getRandomSongs", c.GetRandomSongs)
|
|
h(withPlayer, "getSongsByGenre", c.GetSongsByGenre)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initMediaAnnotationController(api)
|
|
withPlayer := r.With(getPlayer(api.Players))
|
|
h(withPlayer, "setRating", c.SetRating)
|
|
h(withPlayer, "star", c.Star)
|
|
h(withPlayer, "unstar", c.Unstar)
|
|
h(withPlayer, "scrobble", c.Scrobble)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initPlaylistsController(api)
|
|
withPlayer := r.With(getPlayer(api.Players))
|
|
h(withPlayer, "getPlaylists", c.GetPlaylists)
|
|
h(withPlayer, "getPlaylist", c.GetPlaylist)
|
|
h(withPlayer, "createPlaylist", c.CreatePlaylist)
|
|
h(withPlayer, "deletePlaylist", c.DeletePlaylist)
|
|
h(withPlayer, "updatePlaylist", c.UpdatePlaylist)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initBookmarksController(api)
|
|
withPlayer := r.With(getPlayer(api.Players))
|
|
h(withPlayer, "getBookmarks", c.GetBookmarks)
|
|
h(withPlayer, "createBookmark", c.CreateBookmark)
|
|
h(withPlayer, "deleteBookmark", c.DeleteBookmark)
|
|
h(withPlayer, "getPlayQueue", c.GetPlayQueue)
|
|
h(withPlayer, "savePlayQueue", c.SavePlayQueue)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initSearchingController(api)
|
|
withPlayer := r.With(getPlayer(api.Players))
|
|
h(withPlayer, "search2", c.Search2)
|
|
h(withPlayer, "search3", c.Search3)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initUsersController(api)
|
|
h(r, "getUser", c.GetUser)
|
|
h(r, "getUsers", c.GetUsers)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initLibraryScanningController(api)
|
|
h(r, "getScanStatus", c.GetScanStatus)
|
|
h(r, "startScan", c.StartScan)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initMediaRetrievalController(api)
|
|
// configure request throttling
|
|
maxRequests := utils.MaxInt(2, runtime.NumCPU())
|
|
withThrottle := r.With(middleware.ThrottleBacklog(maxRequests, consts.RequestThrottleBacklogLimit, consts.RequestThrottleBacklogTimeout))
|
|
h(withThrottle, "getAvatar", c.GetAvatar)
|
|
h(withThrottle, "getCoverArt", c.GetCoverArt)
|
|
h(withThrottle, "getLyrics", c.GetLyrics)
|
|
})
|
|
r.Group(func(r chi.Router) {
|
|
c := initStreamController(api)
|
|
withPlayer := r.With(getPlayer(api.Players))
|
|
h(withPlayer, "stream", c.Stream)
|
|
h(withPlayer, "download", c.Download)
|
|
})
|
|
|
|
// Not Implemented (yet?)
|
|
h501(r, "jukeboxControl")
|
|
h501(r, "getAlbumInfo", "getAlbumInfo2")
|
|
h501(r, "getShares", "createShare", "updateShare", "deleteShare")
|
|
h501(r, "getPodcasts", "getNewestPodcasts", "refreshPodcasts", "createPodcastChannel", "deletePodcastChannel",
|
|
"deletePodcastEpisode", "downloadPodcastEpisode")
|
|
h501(r, "getInternetRadioStations", "createInternetRadioStation", "updateInternetRadioStation",
|
|
"deleteInternetRadioStation")
|
|
h501(r, "createUser", "updateUser", "deleteUser", "changePassword")
|
|
|
|
// Deprecated/Won't implement/Out of scope endpoints
|
|
h410(r, "search")
|
|
h410(r, "getChatMessages", "addChatMessage")
|
|
h410(r, "getVideos", "getVideoInfo", "getCaptions", "hls")
|
|
return r
|
|
}
|
|
|
|
// Add the Subsonic handler, with and without `.view` extension
|
|
// Ex: if path = `ping` it will create the routes `/ping` and `/ping.view`
|
|
func h(r chi.Router, path string, f handler) {
|
|
handle := func(w http.ResponseWriter, r *http.Request) {
|
|
res, err := f(w, r)
|
|
if err != nil {
|
|
// If it is not a Subsonic error, convert it to an ErrorGeneric
|
|
if _, ok := err.(subError); !ok {
|
|
if err == model.ErrNotFound {
|
|
err = newError(responses.ErrorDataNotFound, "data not found")
|
|
} else {
|
|
err = newError(responses.ErrorGeneric, "Internal Error")
|
|
}
|
|
}
|
|
sendError(w, r, err)
|
|
return
|
|
}
|
|
if res != nil {
|
|
sendResponse(w, r, res)
|
|
}
|
|
}
|
|
r.HandleFunc("/"+path, handle)
|
|
r.HandleFunc("/"+path+".view", handle)
|
|
}
|
|
|
|
// Add a handler that returns 510 - Not implemented. Used to signal that an endpoint is not implemented yet
|
|
func h501(r *chi.Mux, paths ...string) {
|
|
for _, path := range paths {
|
|
handle := func(w http.ResponseWriter, r *http.Request) {
|
|
w.Header().Add("Cache-Control", "no-cache")
|
|
w.WriteHeader(510)
|
|
_, _ = w.Write([]byte("This endpoint is not implemented, but may be in future releases"))
|
|
}
|
|
r.HandleFunc("/"+path, handle)
|
|
r.HandleFunc("/"+path+".view", handle)
|
|
}
|
|
}
|
|
|
|
// Add a handler that returns 410 - Gone. Used to signal that an endpoint will not be implemented
|
|
func h410(r chi.Router, paths ...string) {
|
|
for _, path := range paths {
|
|
handle := func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(410)
|
|
_, _ = w.Write([]byte("This endpoint will not be implemented"))
|
|
}
|
|
r.HandleFunc("/"+path, handle)
|
|
r.HandleFunc("/"+path+".view", handle)
|
|
}
|
|
}
|
|
|
|
func sendError(w http.ResponseWriter, r *http.Request, err error) {
|
|
response := newResponse()
|
|
code := responses.ErrorGeneric
|
|
if e, ok := err.(subError); ok {
|
|
code = e.code
|
|
}
|
|
response.Status = "failed"
|
|
response.Error = &responses.Error{Code: code, Message: err.Error()}
|
|
|
|
sendResponse(w, r, response)
|
|
}
|
|
|
|
func sendResponse(w http.ResponseWriter, r *http.Request, payload *responses.Subsonic) {
|
|
f := utils.ParamString(r, "f")
|
|
var response []byte
|
|
switch f {
|
|
case "json":
|
|
w.Header().Set("Content-Type", "application/json")
|
|
wrapper := &responses.JsonWrapper{Subsonic: *payload}
|
|
response, _ = json.Marshal(wrapper)
|
|
case "jsonp":
|
|
w.Header().Set("Content-Type", "application/javascript")
|
|
callback := utils.ParamString(r, "callback")
|
|
wrapper := &responses.JsonWrapper{Subsonic: *payload}
|
|
data, _ := json.Marshal(wrapper)
|
|
response = []byte(fmt.Sprintf("%s(%s)", callback, data))
|
|
default:
|
|
w.Header().Set("Content-Type", "application/xml")
|
|
response, _ = xml.Marshal(payload)
|
|
}
|
|
if payload.Status == "ok" {
|
|
if log.CurrentLevel() >= log.LevelTrace {
|
|
log.Debug(r.Context(), "API: Successful response", "status", "OK", "body", string(response))
|
|
} else {
|
|
log.Debug(r.Context(), "API: Successful response", "status", "OK")
|
|
}
|
|
} else {
|
|
log.Warn(r.Context(), "API: Failed response", "error", payload.Error.Code, "message", payload.Error.Message)
|
|
}
|
|
if _, err := w.Write(response); err != nil {
|
|
log.Error(r, "Error sending response to client", "payload", string(response), err)
|
|
}
|
|
}
|