package nativeapi import ( "context" "encoding/json" "html" "net/http" "strconv" "time" "github.com/deluan/rest" "github.com/go-chi/chi/v5" "github.com/go-chi/chi/v5/middleware" "github.com/navidrome/navidrome/conf" "github.com/navidrome/navidrome/core" "github.com/navidrome/navidrome/core/metrics" "github.com/navidrome/navidrome/log" "github.com/navidrome/navidrome/model" "github.com/navidrome/navidrome/server" ) type Router struct { http.Handler ds model.DataStore share core.Share playlists core.Playlists insights metrics.Insights } func New(ds model.DataStore, share core.Share, playlists core.Playlists, insights metrics.Insights) *Router { r := &Router{ds: ds, share: share, playlists: playlists, insights: insights} r.Handler = r.routes() return r } func (n *Router) routes() http.Handler { r := chi.NewRouter() // Public n.RX(r, "/translation", newTranslationRepository, false) // Protected r.Group(func(r chi.Router) { r.Use(server.Authenticator(n.ds)) r.Use(server.JWTRefresher) r.Use(server.UpdateLastAccessMiddleware(n.ds)) n.R(r, "/user", model.User{}, true) n.R(r, "/song", model.MediaFile{}, false) n.R(r, "/album", model.Album{}, false) n.R(r, "/artist", model.Artist{}, false) n.R(r, "/genre", model.Genre{}, false) n.R(r, "/player", model.Player{}, true) n.R(r, "/transcoding", model.Transcoding{}, conf.Server.EnableTranscodingConfig) n.R(r, "/radio", model.Radio{}, true) n.R(r, "/tag", model.Tag{}, true) if conf.Server.EnableSharing { n.RX(r, "/share", n.share.NewRepository, true) } n.addPlaylistRoute(r) n.addPlaylistTrackRoute(r) n.addMissingFilesRoute(r) n.addInspectRoute(r) // Keepalive endpoint to be used to keep the session valid (ex: while playing songs) r.Get("/keepalive/*", func(w http.ResponseWriter, r *http.Request) { _, _ = w.Write([]byte(`{"response":"ok", "id":"keepalive"}`)) }) // Insights status endpoint r.Get("/insights/*", func(w http.ResponseWriter, r *http.Request) { last, success := n.insights.LastRun(r.Context()) if conf.Server.EnableInsightsCollector { _, _ = w.Write([]byte(`{"id":"insights_status", "lastRun":"` + last.Format("2006-01-02 15:04:05") + `", "success":` + strconv.FormatBool(success) + `}`)) } else { _, _ = w.Write([]byte(`{"id":"insights_status", "lastRun":"disabled", "success":false}`)) } }) }) return r } func (n *Router) R(r chi.Router, pathPrefix string, model interface{}, persistable bool) { constructor := func(ctx context.Context) rest.Repository { return n.ds.Resource(ctx, model) } n.RX(r, pathPrefix, constructor, persistable) } func (n *Router) RX(r chi.Router, pathPrefix string, constructor rest.RepositoryConstructor, persistable bool) { r.Route(pathPrefix, func(r chi.Router) { r.Get("/", rest.GetAll(constructor)) if persistable { r.Post("/", rest.Post(constructor)) } r.Route("/{id}", func(r chi.Router) { r.Use(server.URLParamsMiddleware) r.Get("/", rest.Get(constructor)) if persistable { r.Put("/", rest.Put(constructor)) r.Delete("/", rest.Delete(constructor)) } }) }) } func (n *Router) addPlaylistRoute(r chi.Router) { constructor := func(ctx context.Context) rest.Repository { return n.ds.Resource(ctx, model.Playlist{}) } r.Route("/playlist", func(r chi.Router) { r.Get("/", rest.GetAll(constructor)) r.Post("/", func(w http.ResponseWriter, r *http.Request) { if r.Header.Get("Content-type") == "application/json" { rest.Post(constructor)(w, r) return } createPlaylistFromM3U(n.playlists)(w, r) }) r.Route("/{id}", func(r chi.Router) { r.Use(server.URLParamsMiddleware) r.Get("/", rest.Get(constructor)) r.Put("/", rest.Put(constructor)) r.Delete("/", rest.Delete(constructor)) }) }) } func (n *Router) addPlaylistTrackRoute(r chi.Router) { r.Route("/playlist/{playlistId}/tracks", func(r chi.Router) { r.Get("/", func(w http.ResponseWriter, r *http.Request) { getPlaylist(n.ds)(w, r) }) r.With(server.URLParamsMiddleware).Route("/", func(r chi.Router) { r.Delete("/", func(w http.ResponseWriter, r *http.Request) { deleteFromPlaylist(n.ds)(w, r) }) r.Post("/", func(w http.ResponseWriter, r *http.Request) { addToPlaylist(n.ds)(w, r) }) }) r.Route("/{id}", func(r chi.Router) { r.Use(server.URLParamsMiddleware) r.Put("/", func(w http.ResponseWriter, r *http.Request) { reorderItem(n.ds)(w, r) }) r.Delete("/", func(w http.ResponseWriter, r *http.Request) { deleteFromPlaylist(n.ds)(w, r) }) }) }) } func (n *Router) addMissingFilesRoute(r chi.Router) { r.Route("/missing", func(r chi.Router) { n.RX(r, "/", newMissingRepository(n.ds), false) r.Delete("/", func(w http.ResponseWriter, r *http.Request) { deleteMissingFiles(n.ds, w, r) }) }) } func writeDeleteManyResponse(w http.ResponseWriter, r *http.Request, ids []string) { var resp []byte var err error if len(ids) == 1 { resp = []byte(`{"id":"` + html.EscapeString(ids[0]) + `"}`) } else { resp, err = json.Marshal(&struct { Ids []string `json:"ids"` }{Ids: ids}) if err != nil { log.Error(r.Context(), "Error marshaling response", "ids", ids, err) http.Error(w, err.Error(), http.StatusInternalServerError) } } _, err = w.Write(resp) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) } } func (n *Router) addInspectRoute(r chi.Router) { if conf.Server.Inspect.Enabled { r.Group(func(r chi.Router) { if conf.Server.Inspect.MaxRequests > 0 { log.Debug("Throttling inspect", "maxRequests", conf.Server.Inspect.MaxRequests, "backlogLimit", conf.Server.Inspect.BacklogLimit, "backlogTimeout", conf.Server.Inspect.BacklogTimeout) r.Use(middleware.ThrottleBacklog(conf.Server.Inspect.MaxRequests, conf.Server.Inspect.BacklogLimit, time.Duration(conf.Server.Inspect.BacklogTimeout))) } r.Get("/inspect", inspect(n.ds)) }) } }