mirror of
https://github.com/owntone/owntone-server.git
synced 2024-12-27 15:45:56 -05:00
[artworkapi/jsonapi] Retrieve artwork for track/album/artist
This commit is contained in:
parent
c1619f7be0
commit
41126d7be4
@ -115,6 +115,7 @@ forked_daapd_SOURCES = main.c \
|
|||||||
httpd_jsonapi.c httpd_jsonapi.h \
|
httpd_jsonapi.c httpd_jsonapi.h \
|
||||||
httpd_streaming.c httpd_streaming.h \
|
httpd_streaming.c httpd_streaming.h \
|
||||||
httpd_oauth.c httpd_oauth.h \
|
httpd_oauth.c httpd_oauth.h \
|
||||||
|
httpd_artworkapi.c httpd_artworkapi.h \
|
||||||
http.c http.h \
|
http.c http.h \
|
||||||
dmap_common.c dmap_common.h \
|
dmap_common.c dmap_common.h \
|
||||||
transcode.h artwork.h \
|
transcode.h artwork.h \
|
||||||
|
16
src/httpd.c
16
src/httpd.c
@ -63,6 +63,7 @@
|
|||||||
#include "httpd_jsonapi.h"
|
#include "httpd_jsonapi.h"
|
||||||
#include "httpd_streaming.h"
|
#include "httpd_streaming.h"
|
||||||
#include "httpd_oauth.h"
|
#include "httpd_oauth.h"
|
||||||
|
#include "httpd_artworkapi.h"
|
||||||
#include "transcode.h"
|
#include "transcode.h"
|
||||||
#ifdef LASTFM
|
#ifdef LASTFM
|
||||||
# include "lastfm.h"
|
# include "lastfm.h"
|
||||||
@ -842,6 +843,11 @@ httpd_gen_cb(struct evhttp_request *req, void *arg)
|
|||||||
jsonapi_request(req, parsed);
|
jsonapi_request(req, parsed);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
else if (artworkapi_is_request(parsed->path))
|
||||||
|
{
|
||||||
|
artworkapi_request(req, parsed);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
else if (streaming_is_request(parsed->path))
|
else if (streaming_is_request(parsed->path))
|
||||||
{
|
{
|
||||||
streaming_request(req, parsed);
|
streaming_request(req, parsed);
|
||||||
@ -1679,6 +1685,14 @@ httpd_init(const char *webroot)
|
|||||||
goto jsonapi_fail;
|
goto jsonapi_fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ret = artworkapi_init();
|
||||||
|
if (ret < 0)
|
||||||
|
{
|
||||||
|
DPRINTF(E_FATAL, L_HTTPD, "Artwork init failed\n");
|
||||||
|
|
||||||
|
goto artworkapi_fail;
|
||||||
|
}
|
||||||
|
|
||||||
ret = oauth_init();
|
ret = oauth_init();
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
@ -1814,6 +1828,8 @@ httpd_init(const char *webroot)
|
|||||||
#endif
|
#endif
|
||||||
oauth_deinit();
|
oauth_deinit();
|
||||||
oauth_fail:
|
oauth_fail:
|
||||||
|
artworkapi_deinit();
|
||||||
|
artworkapi_fail:
|
||||||
jsonapi_deinit();
|
jsonapi_deinit();
|
||||||
jsonapi_fail:
|
jsonapi_fail:
|
||||||
dacp_deinit();
|
dacp_deinit();
|
||||||
|
240
src/httpd_artworkapi.c
Normal file
240
src/httpd_artworkapi.c
Normal file
@ -0,0 +1,240 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2018 Espen Jürgensen <espenjurgensen@gmail.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 <regex.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "httpd_artworkapi.h"
|
||||||
|
#include "logger.h"
|
||||||
|
#include "misc.h"
|
||||||
|
#include "player.h"
|
||||||
|
#include "artwork.h"
|
||||||
|
|
||||||
|
static int
|
||||||
|
request_process(struct httpd_request *hreq, uint32_t *max_w, uint32_t *max_h)
|
||||||
|
{
|
||||||
|
const char *param;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
*max_w = 0;
|
||||||
|
*max_h = 0;
|
||||||
|
|
||||||
|
param = evhttp_find_header(hreq->query, "maxwidth");
|
||||||
|
if (param)
|
||||||
|
{
|
||||||
|
ret = safe_atou32(param, max_w);
|
||||||
|
if (ret < 0)
|
||||||
|
DPRINTF(E_LOG, L_WEB, "Invalid width in request: '%s'\n", hreq->uri_parsed->uri);
|
||||||
|
}
|
||||||
|
|
||||||
|
param = evhttp_find_header(hreq->query, "maxheight");
|
||||||
|
if (param)
|
||||||
|
{
|
||||||
|
ret = safe_atou32(param, max_h);
|
||||||
|
if (ret < 0)
|
||||||
|
DPRINTF(E_LOG, L_WEB, "Invalid height in request: '%s'\n", hreq->uri_parsed->uri);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
response_process(struct httpd_request *hreq, int format)
|
||||||
|
{
|
||||||
|
struct evkeyvalq *headers;
|
||||||
|
|
||||||
|
headers = evhttp_request_get_output_headers(hreq->req);
|
||||||
|
|
||||||
|
if (format == ART_FMT_PNG)
|
||||||
|
evhttp_add_header(headers, "Content-Type", "image/png");
|
||||||
|
else if (format == ART_FMT_JPEG)
|
||||||
|
evhttp_add_header(headers, "Content-Type", "image/jpeg");
|
||||||
|
else
|
||||||
|
return HTTP_NOCONTENT;
|
||||||
|
|
||||||
|
return HTTP_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
artworkapi_reply_nowplaying(struct httpd_request *hreq)
|
||||||
|
{
|
||||||
|
uint32_t max_w;
|
||||||
|
uint32_t max_h;
|
||||||
|
uint32_t id;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = request_process(hreq, &max_w, &max_h);
|
||||||
|
if (ret != 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
ret = player_now_playing(&id);
|
||||||
|
if (ret != 0)
|
||||||
|
return HTTP_NOTFOUND;
|
||||||
|
|
||||||
|
ret = artwork_get_item(hreq->reply, id, max_w, max_h);
|
||||||
|
|
||||||
|
return response_process(hreq, ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
artworkapi_reply_item(struct httpd_request *hreq)
|
||||||
|
{
|
||||||
|
uint32_t max_w;
|
||||||
|
uint32_t max_h;
|
||||||
|
uint32_t id;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = request_process(hreq, &max_w, &max_h);
|
||||||
|
if (ret != 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
ret = safe_atou32(hreq->uri_parsed->path_parts[2], &id);
|
||||||
|
if (ret != 0)
|
||||||
|
return HTTP_BADREQUEST;
|
||||||
|
|
||||||
|
ret = artwork_get_item(hreq->reply, id, max_w, max_h);
|
||||||
|
|
||||||
|
return response_process(hreq, ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
artworkapi_reply_group(struct httpd_request *hreq)
|
||||||
|
{
|
||||||
|
uint32_t max_w;
|
||||||
|
uint32_t max_h;
|
||||||
|
uint32_t id;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = request_process(hreq, &max_w, &max_h);
|
||||||
|
if (ret != 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
ret = safe_atou32(hreq->uri_parsed->path_parts[2], &id);
|
||||||
|
if (ret != 0)
|
||||||
|
return HTTP_BADREQUEST;
|
||||||
|
|
||||||
|
ret = artwork_get_group(hreq->reply, id, max_w, max_h);
|
||||||
|
|
||||||
|
return response_process(hreq, ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct httpd_uri_map artworkapi_handlers[] =
|
||||||
|
{
|
||||||
|
{ EVHTTP_REQ_GET, "^/artwork/nowplaying$", artworkapi_reply_nowplaying },
|
||||||
|
{ EVHTTP_REQ_GET, "^/artwork/item/[[:digit:]]+$", artworkapi_reply_item },
|
||||||
|
{ EVHTTP_REQ_GET, "^/artwork/group/[[:digit:]]+$", artworkapi_reply_group },
|
||||||
|
{ 0, NULL, NULL }
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/* ------------------------------- API --------------------------------- */
|
||||||
|
void
|
||||||
|
artworkapi_request(struct evhttp_request *req, struct httpd_uri_parsed *uri_parsed)
|
||||||
|
{
|
||||||
|
struct httpd_request *hreq;
|
||||||
|
int status_code;
|
||||||
|
|
||||||
|
DPRINTF(E_DBG, L_WEB, "Artwork api request: '%s'\n", uri_parsed->uri);
|
||||||
|
|
||||||
|
if (!httpd_admin_check_auth(req))
|
||||||
|
return;
|
||||||
|
|
||||||
|
hreq = httpd_request_parse(req, uri_parsed, NULL, artworkapi_handlers);
|
||||||
|
if (!hreq)
|
||||||
|
{
|
||||||
|
DPRINTF(E_LOG, L_WEB, "Unrecognized path '%s' in artwork api request: '%s'\n", uri_parsed->path, uri_parsed->uri);
|
||||||
|
|
||||||
|
httpd_send_error(req, HTTP_BADREQUEST, "Bad Request");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
CHECK_NULL(L_WEB, hreq->reply = evbuffer_new());
|
||||||
|
|
||||||
|
status_code = hreq->handler(hreq);
|
||||||
|
|
||||||
|
switch (status_code)
|
||||||
|
{
|
||||||
|
case HTTP_OK: /* 200 OK */
|
||||||
|
httpd_send_reply(req, status_code, "OK", hreq->reply, HTTPD_SEND_NO_GZIP);
|
||||||
|
break;
|
||||||
|
case HTTP_NOCONTENT: /* 204 No Content */
|
||||||
|
httpd_send_reply(req, status_code, "No Content", hreq->reply, HTTPD_SEND_NO_GZIP);
|
||||||
|
break;
|
||||||
|
case HTTP_NOTMODIFIED: /* 304 Not Modified */
|
||||||
|
httpd_send_reply(req, HTTP_NOTMODIFIED, NULL, NULL, HTTPD_SEND_NO_GZIP);
|
||||||
|
break;
|
||||||
|
case HTTP_BADREQUEST: /* 400 Bad Request */
|
||||||
|
httpd_send_error(req, status_code, "Bad Request");
|
||||||
|
break;
|
||||||
|
case HTTP_NOTFOUND: /* 404 Not Found */
|
||||||
|
httpd_send_error(req, status_code, "Not Found");
|
||||||
|
break;
|
||||||
|
case HTTP_INTERNAL: /* 500 Internal Server Error */
|
||||||
|
default:
|
||||||
|
httpd_send_error(req, HTTP_INTERNAL, "Internal Server Error");
|
||||||
|
}
|
||||||
|
|
||||||
|
evbuffer_free(hreq->reply);
|
||||||
|
free(hreq);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
artworkapi_is_request(const char *path)
|
||||||
|
{
|
||||||
|
if (strncmp(path, "/artwork/", strlen("/artwork/")) == 0)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
artworkapi_init(void)
|
||||||
|
{
|
||||||
|
char buf[64];
|
||||||
|
int i;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
for (i = 0; artworkapi_handlers[i].handler; i++)
|
||||||
|
{
|
||||||
|
ret = regcomp(&artworkapi_handlers[i].preg, artworkapi_handlers[i].regexp, REG_EXTENDED | REG_NOSUB);
|
||||||
|
if (ret != 0)
|
||||||
|
{
|
||||||
|
regerror(ret, &artworkapi_handlers[i].preg, buf, sizeof(buf));
|
||||||
|
|
||||||
|
DPRINTF(E_FATAL, L_WEB, "artwork api init failed; regexp error: %s\n", buf);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
artworkapi_deinit(void)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; artworkapi_handlers[i].handler; i++)
|
||||||
|
regfree(&artworkapi_handlers[i].preg);
|
||||||
|
}
|
18
src/httpd_artworkapi.h
Normal file
18
src/httpd_artworkapi.h
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
#ifndef __HTTPD_ARTWORK_H__
|
||||||
|
#define __HTTPD_ARTWORK_H__
|
||||||
|
|
||||||
|
#include "httpd.h"
|
||||||
|
|
||||||
|
int
|
||||||
|
artworkapi_init(void);
|
||||||
|
|
||||||
|
void
|
||||||
|
artworkapi_deinit(void);
|
||||||
|
|
||||||
|
void
|
||||||
|
artworkapi_request(struct evhttp_request *req, struct httpd_uri_parsed *uri_parsed);
|
||||||
|
|
||||||
|
int
|
||||||
|
artworkapi_is_request(const char *path);
|
||||||
|
|
||||||
|
#endif
|
@ -117,6 +117,7 @@ artist_to_json(struct db_group_info *dbgri)
|
|||||||
{
|
{
|
||||||
json_object *item;
|
json_object *item;
|
||||||
char uri[100];
|
char uri[100];
|
||||||
|
char artwork_url[100];
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
item = json_object_new_object();
|
item = json_object_new_object();
|
||||||
@ -132,6 +133,10 @@ artist_to_json(struct db_group_info *dbgri)
|
|||||||
if (ret < sizeof(uri))
|
if (ret < sizeof(uri))
|
||||||
json_object_object_add(item, "uri", json_object_new_string(uri));
|
json_object_object_add(item, "uri", json_object_new_string(uri));
|
||||||
|
|
||||||
|
ret = snprintf(artwork_url, sizeof(artwork_url), "/artwork/group/%s", dbgri->id);
|
||||||
|
if (ret < sizeof(artwork_url))
|
||||||
|
json_object_object_add(item, "artwork_url", json_object_new_string(artwork_url));
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -140,6 +145,7 @@ album_to_json(struct db_group_info *dbgri)
|
|||||||
{
|
{
|
||||||
json_object *item;
|
json_object *item;
|
||||||
char uri[100];
|
char uri[100];
|
||||||
|
char artwork_url[100];
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
item = json_object_new_object();
|
item = json_object_new_object();
|
||||||
@ -156,6 +162,10 @@ album_to_json(struct db_group_info *dbgri)
|
|||||||
if (ret < sizeof(uri))
|
if (ret < sizeof(uri))
|
||||||
json_object_object_add(item, "uri", json_object_new_string(uri));
|
json_object_object_add(item, "uri", json_object_new_string(uri));
|
||||||
|
|
||||||
|
ret = snprintf(artwork_url, sizeof(artwork_url), "/artwork/group/%s", dbgri->id);
|
||||||
|
if (ret < sizeof(artwork_url))
|
||||||
|
json_object_object_add(item, "artwork_url", json_object_new_string(artwork_url));
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -164,6 +174,7 @@ track_to_json(struct db_media_file_info *dbmfi)
|
|||||||
{
|
{
|
||||||
json_object *item;
|
json_object *item;
|
||||||
char uri[100];
|
char uri[100];
|
||||||
|
char artwork_url[100];
|
||||||
int intval;
|
int intval;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
@ -207,6 +218,10 @@ track_to_json(struct db_media_file_info *dbmfi)
|
|||||||
if (ret < sizeof(uri))
|
if (ret < sizeof(uri))
|
||||||
json_object_object_add(item, "uri", json_object_new_string(uri));
|
json_object_object_add(item, "uri", json_object_new_string(uri));
|
||||||
|
|
||||||
|
ret = snprintf(artwork_url, sizeof(artwork_url), "/artwork/item/%s", dbmfi->id);
|
||||||
|
if (ret < sizeof(artwork_url))
|
||||||
|
json_object_object_add(item, "artwork_url", json_object_new_string(artwork_url));
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1466,6 +1481,7 @@ jsonapi_reply_player(struct httpd_request *hreq)
|
|||||||
json_object_object_add(reply, "item_id", json_object_new_int(status.item_id));
|
json_object_object_add(reply, "item_id", json_object_new_int(status.item_id));
|
||||||
json_object_object_add(reply, "item_length_ms", json_object_new_int(status.len_ms));
|
json_object_object_add(reply, "item_length_ms", json_object_new_int(status.len_ms));
|
||||||
json_object_object_add(reply, "item_progress_ms", json_object_new_int(status.pos_ms));
|
json_object_object_add(reply, "item_progress_ms", json_object_new_int(status.pos_ms));
|
||||||
|
json_object_object_add(reply, "artwork_url", json_object_new_string("/artwork/nowplaying"));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -1498,6 +1514,7 @@ queue_item_to_json(struct db_queue_item *queue_item, char shuffle)
|
|||||||
{
|
{
|
||||||
json_object *item;
|
json_object *item;
|
||||||
char uri[100];
|
char uri[100];
|
||||||
|
char artwork_url[100];
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
item = json_object_new_object();
|
item = json_object_new_object();
|
||||||
@ -1534,10 +1551,15 @@ queue_item_to_json(struct db_queue_item *queue_item, char shuffle)
|
|||||||
ret = snprintf(uri, sizeof(uri), "%s:%s:%d", "library", "track", queue_item->file_id);
|
ret = snprintf(uri, sizeof(uri), "%s:%s:%d", "library", "track", queue_item->file_id);
|
||||||
if (ret < sizeof(uri))
|
if (ret < sizeof(uri))
|
||||||
json_object_object_add(item, "uri", json_object_new_string(uri));
|
json_object_object_add(item, "uri", json_object_new_string(uri));
|
||||||
|
|
||||||
|
ret = snprintf(artwork_url, sizeof(artwork_url), "/artwork/item/%d", queue_item->file_id);
|
||||||
|
if (ret < sizeof(artwork_url))
|
||||||
|
json_object_object_add(item, "artwork_url", json_object_new_string(artwork_url));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
safe_json_add_string(item, "uri", queue_item->path);
|
safe_json_add_string(item, "uri", queue_item->path);
|
||||||
|
safe_json_add_string(item, "artwork_url", queue_item->artwork_url);
|
||||||
}
|
}
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
|
Loading…
Reference in New Issue
Block a user