mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-15 16:53:18 -05:00
[pipe/artwork] Support for artwork via Shairport metadata pipes
This implementation uses the cache for storage. Might change that to use a tmpfile instead.
This commit is contained in:
parent
2e149273e5
commit
69fafd873d
@ -138,6 +138,7 @@ static int source_item_cache_get(struct artwork_ctx *ctx);
|
|||||||
static int source_item_embedded_get(struct artwork_ctx *ctx);
|
static int source_item_embedded_get(struct artwork_ctx *ctx);
|
||||||
static int source_item_own_get(struct artwork_ctx *ctx);
|
static int source_item_own_get(struct artwork_ctx *ctx);
|
||||||
static int source_item_stream_get(struct artwork_ctx *ctx);
|
static int source_item_stream_get(struct artwork_ctx *ctx);
|
||||||
|
static int source_item_pipe_get(struct artwork_ctx *ctx);
|
||||||
static int source_item_spotify_get(struct artwork_ctx *ctx);
|
static int source_item_spotify_get(struct artwork_ctx *ctx);
|
||||||
static int source_item_spotifywebapi_get(struct artwork_ctx *ctx);
|
static int source_item_spotifywebapi_get(struct artwork_ctx *ctx);
|
||||||
static int source_item_ownpl_get(struct artwork_ctx *ctx);
|
static int source_item_ownpl_get(struct artwork_ctx *ctx);
|
||||||
@ -196,6 +197,12 @@ static struct artwork_source artwork_item_source[] =
|
|||||||
.data_kinds = (1 << DATA_KIND_HTTP),
|
.data_kinds = (1 << DATA_KIND_HTTP),
|
||||||
.cache = NEVER,
|
.cache = NEVER,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
.name = "pipe",
|
||||||
|
.handler = source_item_pipe_get,
|
||||||
|
.data_kinds = (1 << DATA_KIND_PIPE),
|
||||||
|
.cache = NEVER,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
.name = "Spotify",
|
.name = "Spotify",
|
||||||
.handler = source_item_spotify_get,
|
.handler = source_item_spotify_get,
|
||||||
@ -836,6 +843,26 @@ source_item_stream_get(struct artwork_ctx *ctx)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we are playing a pipe and there is also a metadata pipe, then
|
||||||
|
* input/pipe.c may have saved the incoming artwork via cache_artwork_stash()
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
source_item_pipe_get(struct artwork_ctx *ctx)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
int format;
|
||||||
|
|
||||||
|
DPRINTF(E_SPAM, L_ART, "Trying pipe metadata from %s.metadata\n", ctx->dbmfi->path);
|
||||||
|
|
||||||
|
ret = cache_artwork_read(ctx->evbuf, "pipe://metadata", &format);
|
||||||
|
if (ret < 0)
|
||||||
|
return ART_E_NONE;
|
||||||
|
|
||||||
|
return format;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef HAVE_SPOTIFY_H
|
#ifdef HAVE_SPOTIFY_H
|
||||||
static int
|
static int
|
||||||
source_item_spotify_get(struct artwork_ctx *ctx)
|
source_item_spotify_get(struct artwork_ctx *ctx)
|
||||||
|
@ -62,6 +62,8 @@
|
|||||||
#include "conffile.h"
|
#include "conffile.h"
|
||||||
#include "listener.h"
|
#include "listener.h"
|
||||||
#include "player.h"
|
#include "player.h"
|
||||||
|
#include "artwork.h"
|
||||||
|
#include "cache.h"
|
||||||
#include "worker.h"
|
#include "worker.h"
|
||||||
#include "commands.h"
|
#include "commands.h"
|
||||||
#include "mxml-compat.h"
|
#include "mxml-compat.h"
|
||||||
@ -72,6 +74,8 @@
|
|||||||
#define PIPE_READ_MAX 65536
|
#define PIPE_READ_MAX 65536
|
||||||
// Max number of bytes to buffer from metadata pipes
|
// Max number of bytes to buffer from metadata pipes
|
||||||
#define PIPE_METADATA_BUFLEN_MAX 262144
|
#define PIPE_METADATA_BUFLEN_MAX 262144
|
||||||
|
// Ignore pictures with larger size than this
|
||||||
|
#define PIPE_PICTURE_SIZE_MAX 262144
|
||||||
|
|
||||||
enum pipetype
|
enum pipetype
|
||||||
{
|
{
|
||||||
@ -353,6 +357,34 @@ handle_volume(const char *volume)
|
|||||||
DPRINTF(E_LOG, L_PLAYER, "Shairport airplay volume out of range (-144.0, [-30.0 - 0.0]): %.2f\n", airplay_volume);
|
DPRINTF(E_LOG, L_PLAYER, "Shairport airplay volume out of range (-144.0, [-30.0 - 0.0]): %.2f\n", airplay_volume);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
handle_picture(struct input_metadata *m, uint8_t *data, int data_len)
|
||||||
|
{
|
||||||
|
struct evbuffer *evbuf;
|
||||||
|
int format;
|
||||||
|
|
||||||
|
if (data_len < 2 || data_len > PIPE_PICTURE_SIZE_MAX)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
if (data[0] == 0xff && data[1] == 0xd8)
|
||||||
|
format = ART_FMT_JPEG;
|
||||||
|
else if (data[0] == 0x89 && data[1] == 0x50)
|
||||||
|
format = ART_FMT_PNG;
|
||||||
|
else
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
CHECK_NULL(L_PLAYER, evbuf = evbuffer_new());
|
||||||
|
|
||||||
|
evbuffer_add(evbuf, data, data_len);
|
||||||
|
cache_artwork_stash(evbuf, "pipe://metadata", format);
|
||||||
|
evbuffer_free(evbuf);
|
||||||
|
return;
|
||||||
|
|
||||||
|
error:
|
||||||
|
DPRINTF(E_LOG, L_PLAYER, "Unsupported picture size (%d) or format from Shairport metadata pipe\n", data_len);
|
||||||
|
cache_artwork_stash(NULL, NULL, 0); // Clear the artwork stash
|
||||||
|
}
|
||||||
|
|
||||||
// returns 1 on metadata found, 0 on nothing, -1 on error
|
// returns 1 on metadata found, 0 on nothing, -1 on error
|
||||||
static int
|
static int
|
||||||
handle_item(struct input_metadata *m, const char *item)
|
handle_item(struct input_metadata *m, const char *item)
|
||||||
@ -366,6 +398,7 @@ handle_item(struct input_metadata *m, const char *item)
|
|||||||
char **dstptr;
|
char **dstptr;
|
||||||
char *progress;
|
char *progress;
|
||||||
char *volume;
|
char *volume;
|
||||||
|
char *picture;
|
||||||
uint8_t *data;
|
uint8_t *data;
|
||||||
int data_len;
|
int data_len;
|
||||||
int ret;
|
int ret;
|
||||||
@ -412,6 +445,8 @@ handle_item(struct input_metadata *m, const char *item)
|
|||||||
dstptr = &progress;
|
dstptr = &progress;
|
||||||
else if (code == dmapval("pvol"))
|
else if (code == dmapval("pvol"))
|
||||||
dstptr = &volume;
|
dstptr = &volume;
|
||||||
|
else if (code == dmapval("PICT"))
|
||||||
|
dstptr = &picture;
|
||||||
else
|
else
|
||||||
goto out_nothing;
|
goto out_nothing;
|
||||||
|
|
||||||
@ -429,7 +464,7 @@ handle_item(struct input_metadata *m, const char *item)
|
|||||||
goto out_error;
|
goto out_error;
|
||||||
}
|
}
|
||||||
|
|
||||||
DPRINTF(E_DBG, L_PLAYER, "Read Shairport metadata (type=%8x, code=%8x, len=%d): '%s'\n", type, code, data_len, (char *)data);
|
DPRINTF(E_DBG, L_PLAYER, "Read Shairport metadata (type=%8x, code=%8x, len=%d)\n", type, code, data_len);
|
||||||
|
|
||||||
if (dstptr == &progress)
|
if (dstptr == &progress)
|
||||||
{
|
{
|
||||||
@ -443,6 +478,11 @@ handle_item(struct input_metadata *m, const char *item)
|
|||||||
handle_volume(volume);
|
handle_volume(volume);
|
||||||
free(data);
|
free(data);
|
||||||
}
|
}
|
||||||
|
else if (dstptr == &picture)
|
||||||
|
{
|
||||||
|
handle_picture(m, data, data_len);
|
||||||
|
free(data);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
free(*dstptr); // If m->x is already set, free it
|
free(*dstptr); // If m->x is already set, free it
|
||||||
|
Loading…
x
Reference in New Issue
Block a user