mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-15 00:35:03 -05:00
2778088c52
Removes a lot of code duplication which hopefully will make it easier to add support for evhtp as http engine.
208 lines
5.2 KiB
C
208 lines
5.2 KiB
C
/*
|
|
* 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_internal.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_playing_now(&id);
|
|
if (ret != 0)
|
|
return HTTP_NOTFOUND;
|
|
|
|
ret = artwork_get_item(hreq->reply, id, max_w, max_h, 0);
|
|
|
|
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, 0);
|
|
|
|
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, 0);
|
|
|
|
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 --------------------------------- */
|
|
|
|
static void
|
|
artworkapi_request(struct httpd_request *hreq)
|
|
{
|
|
int status_code;
|
|
|
|
DPRINTF(E_DBG, L_WEB, "Artwork api request: '%s'\n", hreq->uri);
|
|
|
|
if (!httpd_admin_check_auth(hreq->req))
|
|
return;
|
|
|
|
if (!hreq->handler)
|
|
{
|
|
DPRINTF(E_LOG, L_WEB, "Unrecognized path in artwork api request: '%s'\n", hreq->uri);
|
|
|
|
httpd_send_error(hreq->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(hreq->req, status_code, "OK", hreq->reply, HTTPD_SEND_NO_GZIP);
|
|
break;
|
|
case HTTP_NOCONTENT: /* 204 No Content */
|
|
httpd_send_reply(hreq->req, status_code, "No Content", hreq->reply, HTTPD_SEND_NO_GZIP);
|
|
break;
|
|
case HTTP_NOTMODIFIED: /* 304 Not Modified */
|
|
httpd_send_reply(hreq->req, HTTP_NOTMODIFIED, NULL, NULL, HTTPD_SEND_NO_GZIP);
|
|
break;
|
|
case HTTP_BADREQUEST: /* 400 Bad Request */
|
|
httpd_send_error(hreq->req, status_code, "Bad Request");
|
|
break;
|
|
case HTTP_NOTFOUND: /* 404 Not Found */
|
|
httpd_send_error(hreq->req, status_code, "Not Found");
|
|
break;
|
|
case HTTP_INTERNAL: /* 500 Internal Server Error */
|
|
default:
|
|
httpd_send_error(hreq->req, HTTP_INTERNAL, "Internal Server Error");
|
|
}
|
|
|
|
evbuffer_free(hreq->reply);
|
|
}
|
|
|
|
struct httpd_module httpd_artworkapi =
|
|
{
|
|
.name = "Artwork API",
|
|
.type = MODULE_ARTWORKAPI,
|
|
.subpaths = { "/artwork/", NULL },
|
|
.handlers = artworkapi_handlers,
|
|
.request = artworkapi_request,
|
|
};
|