owntone-server/src/httpd_jsonapi.c

658 lines
17 KiB
C
Raw Normal View History

2017-08-12 13:42:06 -04:00
/*
* Copyright (C) 2017 Christian Meffert <christian.meffert@googlemail.com>
*
* Adapted from httpd_adm.c:
* Copyright (C) 2015 Stuart NAIFEH <stu@naifeh.org>
*
* Adapted from httpd_daap.c and httpd.c:
* Copyright (C) 2009-2011 Julien BLACHE <jb@jblache.org>
* Copyright (C) 2010 Kai Elwert <elwertk@googlemail.com>
*
* Adapted from mt-daapd:
* Copyright (C) 2003-2007 Ron Pedde <ron@pedde.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include <json.h>
#include <regex.h>
#include <string.h>
#include "httpd_jsonapi.h"
2017-08-12 13:42:06 -04:00
#include "conffile.h"
#include "db.h"
#ifdef LASTFM
# include "lastfm.h"
#endif
2017-08-12 13:42:06 -04:00
#include "library.h"
#include "logger.h"
2017-09-15 18:10:26 -04:00
#include "misc.h"
2017-08-12 13:42:06 -04:00
#include "misc_json.h"
#include "remote_pairing.h"
#ifdef HAVE_SPOTIFY_H
# include "spotify_webapi.h"
# include "spotify.h"
#endif
struct json_request {
// The parsed request URI given to us by httpd.c
struct httpd_uri_parsed *uri_parsed;
// Shortcut to &uri_parsed->ev_query
struct evkeyvalq *query;
// http request struct
struct evhttp_request *req;
// A pointer to the handler that will process the request
int (*handler)(struct evbuffer *reply, struct json_request *jreq);
};
2017-08-12 13:42:06 -04:00
struct uri_map
{
regex_t preg;
char *regexp;
int (*handler)(struct evbuffer *reply, struct json_request *jreq);
2017-08-12 13:42:06 -04:00
};
/* Forward declaration of handlers */
static struct uri_map adm_handlers[];
/* -------------------------------- HELPERS --------------------------------- */
/*
* Kicks off pairing of a daap/dacp client
*
* Expects the paring pin to be present in the post request body, e. g.:
*
* {
* "pin": "1234"
* }
*/
static int
pairing_kickoff(struct evhttp_request *req)
{
struct evbuffer *evbuf;
json_object* request;
const char* message;
evbuf = evhttp_request_get_input_buffer(req);
request = jparse_obj_from_evbuffer(evbuf);
if (!request)
{
DPRINTF(E_LOG, L_WEB, "Failed to parse incoming request\n");
return -1;
}
DPRINTF(E_DBG, L_WEB, "Received pairing post request: %s\n", json_object_to_json_string(request));
message = jparse_str_from_obj(request, "pin");
if (message)
remote_pairing_kickoff((char **)&message);
else
DPRINTF(E_LOG, L_WEB, "Missing pin in request body: %s\n", json_object_to_json_string(request));
jparse_free(request);
return 0;
}
/*
* Retrieves pairing information
*
* Example response:
*
* {
* "active": true,
* "remote": "remote name"
* }
*/
static int
pairing_get(struct evbuffer *evbuf)
{
char *remote_name;
json_object *jreply;
remote_name = remote_pairing_get_name();
CHECK_NULL(L_WEB, jreply = json_object_new_object());
if (remote_name)
{
json_object_object_add(jreply, "active", json_object_new_boolean(true));
json_object_object_add(jreply, "remote", json_object_new_string(remote_name));
}
else
{
json_object_object_add(jreply, "active", json_object_new_boolean(false));
}
CHECK_ERRNO(L_WEB, evbuffer_add_printf(evbuf, "%s", json_object_to_json_string(jreply)));
jparse_free(jreply);
free(remote_name);
return 0;
}
static struct json_request *
json_request_parse(struct evhttp_request *req, struct httpd_uri_parsed *uri_parsed)
{
struct json_request *jreq;
int ret;
int i;
2017-08-12 13:42:06 -04:00
CHECK_NULL(L_WEB, jreq = calloc(1, sizeof(struct json_request)));
jreq->req = req;
jreq->uri_parsed = uri_parsed;
jreq->query = &(uri_parsed->ev_query);
// Find a handler for the path
for (i = 0; adm_handlers[i].handler; i++)
{
ret = regexec(&adm_handlers[i].preg, uri_parsed->path, 0, NULL, 0);
if (ret == 0)
{
jreq->handler = adm_handlers[i].handler;
break;
}
}
if (!jreq->handler)
{
DPRINTF(E_LOG, L_WEB, "Unrecognized path '%s' in JSON api request: '%s'\n", uri_parsed->path, uri_parsed->uri);
goto error;
}
return jreq;
error:
free(jreq);
return NULL;
}
/* --------------------------- REPLY HANDLERS ------------------------------- */
2017-08-12 13:42:06 -04:00
/*
* Endpoint to retrieve configuration values
*
* Example response:
*
* {
* "websocket_port": 6603,
* "version": "25.0"
* }
*/
static int
jsonapi_reply_config(struct evbuffer *reply, struct json_request *jreq)
2017-08-12 13:42:06 -04:00
{
json_object *jreply;
2017-08-12 13:42:06 -04:00
json_object *buildopts;
int websocket_port;
char **buildoptions;
int i;
CHECK_NULL(L_WEB, jreply = json_object_new_object());
2017-08-12 13:42:06 -04:00
// Websocket port
#ifdef HAVE_LIBWEBSOCKETS
2017-08-12 13:42:06 -04:00
websocket_port = cfg_getint(cfg_getsec(cfg, "general"), "websocket_port");
#else
websocket_port = 0;
#endif
json_object_object_add(jreply, "websocket_port", json_object_new_int(websocket_port));
2017-08-12 13:42:06 -04:00
// forked-daapd version
json_object_object_add(jreply, "version", json_object_new_string(VERSION));
2017-08-12 13:42:06 -04:00
// enabled build options
buildopts = json_object_new_array();
buildoptions = buildopts_get();
for (i = 0; buildoptions[i]; i++)
{
json_object_array_add(buildopts, json_object_new_string(buildoptions[i]));
}
json_object_object_add(jreply, "buildoptions", buildopts);
2017-08-12 13:42:06 -04:00
CHECK_ERRNO(L_WEB, evbuffer_add_printf(reply, "%s", json_object_to_json_string(jreply)));
jparse_free(jreply);
2017-08-12 13:42:06 -04:00
return 0;
}
/*
* Endpoint to retrieve informations about the library
*
* Example response:
*
* {
* "artists": 84,
* "albums": 151,
* "songs": 3085,
* "db_playtime": 687824,
* "updating": false
*}
*/
static int
jsonapi_reply_library(struct evbuffer *reply, struct json_request *jreq)
2017-08-12 13:42:06 -04:00
{
struct query_params qp;
struct filecount_info fci;
int artists;
int albums;
bool is_scanning;
json_object *jreply;
2017-08-12 13:42:06 -04:00
int ret;
// Fetch values for response
memset(&qp, 0, sizeof(struct query_params));
qp.type = Q_COUNT_ITEMS;
ret = db_filecount_get(&fci, &qp);
if (ret < 0)
{
DPRINTF(E_LOG, L_WEB, "library: failed to get file count info\n");
return -1;
}
artists = db_files_get_artist_count();
albums = db_files_get_album_count();
is_scanning = library_is_scanning();
// Build json response
CHECK_NULL(L_WEB, jreply = json_object_new_object());
2017-08-12 13:42:06 -04:00
json_object_object_add(jreply, "artists", json_object_new_int(artists));
json_object_object_add(jreply, "albums", json_object_new_int(albums));
json_object_object_add(jreply, "songs", json_object_new_int(fci.count));
json_object_object_add(jreply, "db_playtime", json_object_new_int64((fci.length / 1000)));
json_object_object_add(jreply, "updating", json_object_new_boolean(is_scanning));
2017-08-12 13:42:06 -04:00
CHECK_ERRNO(L_WEB, evbuffer_add_printf(reply, "%s", json_object_to_json_string(jreply)));
jparse_free(jreply);
2017-08-12 13:42:06 -04:00
return 0;
}
/*
* Endpoint to trigger a library rescan
*/
static int
jsonapi_reply_update(struct evbuffer *reply, struct json_request *jreq)
2017-08-12 13:42:06 -04:00
{
library_rescan();
return 0;
}
/*
* Endpoint to retrieve information about the spotify integration
*
* Exampe response:
*
* {
* "enabled": true,
* "oauth_uri": "https://accounts.spotify.com/authorize/?client_id=...
* }
*/
static int
jsonapi_reply_spotify(struct evbuffer *reply, struct json_request *jreq)
2017-08-12 13:42:06 -04:00
{
json_object *jreply;
2017-08-12 13:42:06 -04:00
CHECK_NULL(L_WEB, jreply = json_object_new_object());
2017-08-12 13:42:06 -04:00
#ifdef HAVE_SPOTIFY_H
int httpd_port;
char redirect_uri[256];
char *oauth_uri;
2017-08-12 13:42:06 -04:00
struct spotify_status_info info;
json_object_object_add(jreply, "enabled", json_object_new_boolean(true));
2017-08-12 13:42:06 -04:00
httpd_port = cfg_getint(cfg_getsec(cfg, "library"), "port");
snprintf(redirect_uri, sizeof(redirect_uri), "http://forked-daapd.local:%d/oauth/spotify", httpd_port);
oauth_uri = spotifywebapi_oauth_uri_get(redirect_uri);
if (!oauth_uri)
2017-08-12 13:42:06 -04:00
{
DPRINTF(E_LOG, L_WEB, "Cannot display Spotify oauth interface (http_form_uriencode() failed)\n");
jparse_free(jreply);
return -1;
2017-08-12 13:42:06 -04:00
}
json_object_object_add(jreply, "oauth_uri", json_object_new_string(oauth_uri));
free(oauth_uri);
2017-08-12 13:42:06 -04:00
spotify_status_info_get(&info);
json_object_object_add(jreply, "libspotify_installed", json_object_new_boolean(info.libspotify_installed));
json_object_object_add(jreply, "libspotify_logged_in", json_object_new_boolean(info.libspotify_logged_in));
json_object_object_add(jreply, "libspotify_user", json_object_new_string(info.libspotify_user));
json_object_object_add(jreply, "webapi_token_valid", json_object_new_boolean(info.webapi_token_valid));
json_object_object_add(jreply, "webapi_user", json_object_new_string(info.webapi_user));
2017-08-12 13:42:06 -04:00
#else
json_object_object_add(jreply, "enabled", json_object_new_boolean(false));
2017-08-12 13:42:06 -04:00
#endif
CHECK_ERRNO(L_WEB, evbuffer_add_printf(reply, "%s", json_object_to_json_string(jreply)));
jparse_free(jreply);
2017-08-12 13:42:06 -04:00
return 0;
}
static int
jsonapi_reply_spotify_login(struct evbuffer *reply, struct json_request *jreq)
2017-08-12 13:42:06 -04:00
{
#ifdef HAVE_SPOTIFY_H
2017-08-12 13:42:06 -04:00
struct evbuffer *in_evbuf;
json_object* request;
const char *user;
const char *password;
char *errmsg = NULL;
json_object* jreply;
2017-08-12 13:42:06 -04:00
json_object* errors;
int ret;
DPRINTF(E_DBG, L_WEB, "Received Spotify login request\n");
in_evbuf = evhttp_request_get_input_buffer(jreq->req);
2017-08-12 13:42:06 -04:00
request = jparse_obj_from_evbuffer(in_evbuf);
if (!request)
{
DPRINTF(E_LOG, L_WEB, "Failed to parse incoming request\n");
return -1;
}
CHECK_NULL(L_WEB, jreply = json_object_new_object());
2017-08-12 13:42:06 -04:00
user = jparse_str_from_obj(request, "user");
password = jparse_str_from_obj(request, "password");
if (user && strlen(user) > 0 && password && strlen(password) > 0)
{
ret = spotify_login_user(user, password, &errmsg);
if (ret < 0)
{
json_object_object_add(jreply, "success", json_object_new_boolean(false));
2017-08-12 13:42:06 -04:00
errors = json_object_new_object();
json_object_object_add(errors, "error", json_object_new_string(errmsg));
json_object_object_add(jreply, "errors", errors);
2017-08-12 13:42:06 -04:00
}
else
{
json_object_object_add(jreply, "success", json_object_new_boolean(true));
2017-08-12 13:42:06 -04:00
}
free(errmsg);
}
else
{
DPRINTF(E_LOG, L_WEB, "No user or password in spotify login post request\n");
json_object_object_add(jreply, "success", json_object_new_boolean(false));
2017-08-12 13:42:06 -04:00
errors = json_object_new_object();
if (!user || strlen(user) == 0)
json_object_object_add(errors, "user", json_object_new_string("Username is required"));
if (!password || strlen(password) == 0)
json_object_object_add(errors, "password", json_object_new_string("Password is required"));
json_object_object_add(jreply, "errors", errors);
2017-08-12 13:42:06 -04:00
}
CHECK_ERRNO(L_WEB, evbuffer_add_printf(reply, "%s", json_object_to_json_string(jreply)));
jparse_free(jreply);
2017-08-12 13:42:06 -04:00
#else
DPRINTF(E_LOG, L_WEB, "Received spotify login request but was not compiled with enable-spotify\n");
#endif
return 0;
}
/*
* Endpoint to pair daap/dacp client
*
* If request is a GET request, returns information about active pairing remote.
* If request is a POST request, tries to pair the active remote with the given pin.
*/
static int
jsonapi_reply_pairing(struct evbuffer *reply, struct json_request *jreq)
2017-08-12 13:42:06 -04:00
{
if (evhttp_request_get_command(jreq->req) == EVHTTP_REQ_POST)
2017-08-12 13:42:06 -04:00
{
return pairing_kickoff(jreq->req);
2017-08-12 13:42:06 -04:00
}
return pairing_get(reply);
2017-08-12 13:42:06 -04:00
}
static int
jsonapi_reply_lastfm(struct evbuffer *reply, struct json_request *jreq)
{
json_object *jreply;
bool enabled = false;
bool scrobbling_enabled = false;
#ifdef LASTFM
enabled = true;
scrobbling_enabled = lastfm_is_enabled();
#endif
CHECK_NULL(L_WEB, jreply = json_object_new_object());
json_object_object_add(jreply, "enabled", json_object_new_boolean(enabled));
json_object_object_add(jreply, "scrobbling_enabled", json_object_new_boolean(scrobbling_enabled));
CHECK_ERRNO(L_WEB, evbuffer_add_printf(reply, "%s", json_object_to_json_string(jreply)));
jparse_free(jreply);
return 0;
}
/*
* Endpoint to log into LastFM
*/
static int
jsonapi_reply_lastfm_login(struct evbuffer *reply, struct json_request *jreq)
{
#ifdef LASTFM
struct evbuffer *in_evbuf;
json_object *request;
const char *user;
const char *password;
char *errmsg = NULL;
json_object *jreply;
json_object *errors;
int ret;
DPRINTF(E_DBG, L_WEB, "Received LastFM login request\n");
in_evbuf = evhttp_request_get_input_buffer(jreq->req);
request = jparse_obj_from_evbuffer(in_evbuf);
if (!request)
{
DPRINTF(E_LOG, L_WEB, "Failed to parse incoming request\n");
return -1;
}
CHECK_NULL(L_WEB, jreply = json_object_new_object());
user = jparse_str_from_obj(request, "user");
password = jparse_str_from_obj(request, "password");
if (user && strlen(user) > 0 && password && strlen(password) > 0)
{
ret = lastfm_login_user(user, password, &errmsg);
if (ret < 0)
{
json_object_object_add(jreply, "success", json_object_new_boolean(false));
errors = json_object_new_object();
if (errmsg)
json_object_object_add(errors, "error", json_object_new_string(errmsg));
else
json_object_object_add(errors, "error", json_object_new_string("Unknown error"));
json_object_object_add(jreply, "errors", errors);
}
else
{
json_object_object_add(jreply, "success", json_object_new_boolean(true));
}
free(errmsg);
}
else
{
DPRINTF(E_LOG, L_WEB, "No user or password in LastFM login post request\n");
json_object_object_add(jreply, "success", json_object_new_boolean(false));
errors = json_object_new_object();
if (!user || strlen(user) == 0)
json_object_object_add(errors, "user", json_object_new_string("Username is required"));
if (!password || strlen(password) == 0)
json_object_object_add(errors, "password", json_object_new_string("Password is required"));
json_object_object_add(jreply, "errors", errors);
}
CHECK_ERRNO(L_WEB, evbuffer_add_printf(reply, "%s", json_object_to_json_string(jreply)));
jparse_free(jreply);
#else
DPRINTF(E_LOG, L_WEB, "Received LastFM login request but was not compiled with enable-lastfm\n");
#endif
return 0;
}
static int
jsonapi_reply_lastfm_logout(struct evbuffer *reply, struct json_request *jreq)
{
#ifdef LASTFM
lastfm_logout();
#endif
return 0;
}
2017-08-12 13:42:06 -04:00
static struct uri_map adm_handlers[] =
{
{ .regexp = "^/api/config", .handler = jsonapi_reply_config },
{ .regexp = "^/api/library", .handler = jsonapi_reply_library },
{ .regexp = "^/api/update", .handler = jsonapi_reply_update },
{ .regexp = "^/api/spotify-login", .handler = jsonapi_reply_spotify_login },
{ .regexp = "^/api/spotify", .handler = jsonapi_reply_spotify },
{ .regexp = "^/api/pairing", .handler = jsonapi_reply_pairing },
{ .regexp = "^/api/lastfm-login", .handler = jsonapi_reply_lastfm_login },
{ .regexp = "^/api/lastfm-logout", .handler = jsonapi_reply_lastfm_logout },
{ .regexp = "^/api/lastfm", .handler = jsonapi_reply_lastfm },
2017-08-12 13:42:06 -04:00
{ .regexp = NULL, .handler = NULL }
};
/* ------------------------------- JSON API --------------------------------- */
2017-08-12 13:42:06 -04:00
void
jsonapi_request(struct evhttp_request *req, struct httpd_uri_parsed *uri_parsed)
2017-08-12 13:42:06 -04:00
{
struct json_request *jreq;
struct evbuffer *reply;
2017-08-12 13:42:06 -04:00
struct evkeyvalq *headers;
int ret;
DPRINTF(E_DBG, L_WEB, "JSON api request: '%s'\n", uri_parsed->uri);
2017-08-12 13:42:06 -04:00
if (!httpd_admin_check_auth(req))
2017-08-12 13:42:06 -04:00
{
DPRINTF(E_DBG, L_WEB, "JSON api request denied\n");
2017-08-12 13:42:06 -04:00
return;
}
jreq = json_request_parse(req, uri_parsed);
if (!jreq)
2017-08-12 13:42:06 -04:00
{
httpd_send_error(req, HTTP_BADREQUEST, "Bad Request");
2017-08-12 13:42:06 -04:00
return;
}
headers = evhttp_request_get_output_headers(req);
evhttp_add_header(headers, "DAAP-Server", "forked-daapd/" VERSION);
CHECK_NULL(L_WEB, reply = evbuffer_new());
ret = jreq->handler(reply, jreq);
2017-08-12 13:42:06 -04:00
if (ret < 0)
{
httpd_send_error(req, 500, "Internal Server Error");
goto error;
2017-08-12 13:42:06 -04:00
}
evhttp_add_header(headers, "Content-Type", "application/json");
httpd_send_reply(req, HTTP_OK, "OK", reply, 0);
error:
evbuffer_free(reply);
free(jreq);
2017-08-12 13:42:06 -04:00
}
int
jsonapi_is_request(const char *path)
2017-08-12 13:42:06 -04:00
{
if (strncmp(path, "/api/", strlen("/api/")) == 0)
2017-08-12 13:42:06 -04:00
return 1;
if (strcmp(path, "/api") == 0)
2017-08-12 13:42:06 -04:00
return 1;
return 0;
}
int
jsonapi_init(void)
{
char buf[64];
int i;
int ret;
for (i = 0; adm_handlers[i].handler; i++)
{
ret = regcomp(&adm_handlers[i].preg, adm_handlers[i].regexp, REG_EXTENDED | REG_NOSUB);
if (ret != 0)
{
regerror(ret, &adm_handlers[i].preg, buf, sizeof(buf));
DPRINTF(E_FATAL, L_WEB, "JSON api init failed; regexp error: %s\n", buf);
2017-08-12 13:42:06 -04:00
return -1;
}
}
return 0;
}
void
jsonapi_deinit(void)
{
int i;
for (i = 0; adm_handlers[i].handler; i++)
regfree(&adm_handlers[i].preg);
}