Implement is_remote() and change how transcode_needed() is used

transcode_needed() was getting called needlessly in http_daapd.c,
because 1) once it is determined that a given codec needs transcoding
for a given client there is no reason to call and check again, 2)
transcoding is irrelevant for remotes. Also some cleaning up of
user_agent_filter().
This commit is contained in:
ejurgensen 2014-12-28 23:37:12 +01:00
parent 3e412b5e65
commit a69619a5a7
2 changed files with 55 additions and 27 deletions

View File

@ -506,14 +506,31 @@ daap_sort_finalize(struct sort_ctx *ctx)
dmap_add_int(ctx->headerlist, "mshn", ctx->misc_mshn); /* 12 */ dmap_add_int(ctx->headerlist, "mshn", ctx->misc_mshn); /* 12 */
} }
/* Remotes are clients that will issue DACP commands. For these clients we will
* do the playback, and we will not stream to them. This is a crude function to
* identify them, so we can give them appropriate treatment.
*/
static int
is_remote(const char *user_agent)
{
if (!user_agent)
return 0;
if (strcasestr(user_agent, "remote"))
return 1;
if (strstr(user_agent, "Retune"))
return 1;
return 0;
}
/* We try not to return items that the client cannot play (like Spotify and /* We try not to return items that the client cannot play (like Spotify and
* internet streams in iTunes), or which are inappropriate (like internet streams * internet streams in iTunes), or which are inappropriate (like internet streams
* in the album tab in Remote * in the album tab of remotes)
*/ */
static void static void
user_agent_filter(const char *user_agent, struct query_params *qp) user_agent_filter(const char *user_agent, struct query_params *qp)
{ {
char *filter; const char *filter;
char *buffer; char *buffer;
int len; int len;
@ -523,16 +540,10 @@ user_agent_filter(const char *user_agent, struct query_params *qp)
// Valgrind doesn't like strlen(filter) below, so instead we allocate 128 bytes // Valgrind doesn't like strlen(filter) below, so instead we allocate 128 bytes
// to hold the string and the leading " AND ". Remember to adjust the 128 if // to hold the string and the leading " AND ". Remember to adjust the 128 if
// you define strings here that will be too large for the buffer. // you define strings here that will be too large for the buffer.
if (strcasestr(user_agent, "itunes")) if (is_remote(user_agent))
filter = strdup("(f.data_kind = 0)"); // Only real files filter = "(f.data_kind <> 1)"; // No internet radio
else if (strcasestr(user_agent, "daap"))
filter = strdup("(f.data_kind = 0)"); // Only real files
else if (strcasestr(user_agent, "remote"))
filter = strdup("(f.data_kind <> 1)"); // No internet radio
else if (strcasestr(user_agent, "android"))
filter = strdup("(f.data_kind <> 1)"); // No internet radio
else else
return; filter = "(f.data_kind = 0)"; // Only real files
if (qp->filter) if (qp->filter)
{ {
@ -547,8 +558,6 @@ user_agent_filter(const char *user_agent, struct query_params *qp)
qp->filter = strdup(filter); qp->filter = strdup(filter);
DPRINTF(E_DBG, L_DAAP, "SQL filter w/client mod: %s\n", qp->filter); DPRINTF(E_DBG, L_DAAP, "SQL filter w/client mod: %s\n", qp->filter);
free(filter);
} }
/* Returns eg /databases/1/containers from /databases/1/containers?meta=dmap.item... */ /* Returns eg /databases/1/containers from /databases/1/containers?meta=dmap.item... */
@ -1169,10 +1178,12 @@ daap_reply_songlist_generic(struct evhttp_request *req, struct evbuffer *evbuf,
struct sort_ctx *sctx; struct sort_ctx *sctx;
const char *param; const char *param;
const char *client_codecs; const char *client_codecs;
char *last_codectype;
char *tag; char *tag;
int nmeta; int nmeta;
int sort_headers; int sort_headers;
int nsongs; int nsongs;
int remote;
int transcode; int transcode;
int ret; int ret;
@ -1299,20 +1310,41 @@ daap_reply_songlist_generic(struct evhttp_request *req, struct evbuffer *evbuf,
goto out_query_free; goto out_query_free;
} }
nsongs = 0; remote = is_remote(ua);
while (((ret = db_query_fetch_file(&qp, &dbmfi)) == 0) && (dbmfi.id))
{
nsongs++;
client_codecs = NULL; client_codecs = NULL;
if (req) if (!remote && req)
{ {
headers = evhttp_request_get_input_headers(req); headers = evhttp_request_get_input_headers(req);
client_codecs = evhttp_find_header(headers, "Accept-Codecs"); client_codecs = evhttp_find_header(headers, "Accept-Codecs");
} }
nsongs = 0;
last_codectype = NULL;
while (((ret = db_query_fetch_file(&qp, &dbmfi)) == 0) && (dbmfi.id))
{
nsongs++;
if (!dbmfi.codectype)
{
DPRINTF(E_LOG, L_DAAP, "Cannot transcode '%s', codec type is unknown\n", dbmfi.fname);
transcode = 0;
}
else if (remote)
{
transcode = 1;
}
else if (!last_codectype || (strcmp(last_codectype, dbmfi.codectype) != 0))
{
transcode = transcode_needed(ua, client_codecs, dbmfi.codectype); transcode = transcode_needed(ua, client_codecs, dbmfi.codectype);
if (last_codectype)
free(last_codectype);
last_codectype = strdup(dbmfi.codectype);
}
ret = dmap_encode_file_metadata(songlist, song, &dbmfi, meta, nmeta, sort_headers, transcode); ret = dmap_encode_file_metadata(songlist, song, &dbmfi, meta, nmeta, sort_headers, transcode);
if (ret < 0) if (ret < 0)
{ {
@ -1339,6 +1371,9 @@ daap_reply_songlist_generic(struct evhttp_request *req, struct evbuffer *evbuf,
DPRINTF(E_DBG, L_DAAP, "Done with song list, %d songs\n", nsongs); DPRINTF(E_DBG, L_DAAP, "Done with song list, %d songs\n", nsongs);
if (last_codectype)
free(last_codectype);
if (nmeta > 0) if (nmeta > 0)
free(meta); free(meta);

View File

@ -767,7 +767,6 @@ transcode_cleanup(struct transcode_ctx *ctx)
free(ctx); free(ctx);
} }
int int
transcode_needed(const char *user_agent, const char *client_codecs, char *file_codectype) transcode_needed(const char *user_agent, const char *client_codecs, char *file_codectype)
{ {
@ -776,15 +775,9 @@ transcode_needed(const char *user_agent, const char *client_codecs, char *file_c
int size; int size;
int i; int i;
// If client is a Remote we will AirPlay, which means we will transcode to PCM
if (user_agent && strcasestr(user_agent, "remote"))
return 1;
else if (user_agent && strcasestr(user_agent, "android"))
return 1;
if (!file_codectype) if (!file_codectype)
{ {
DPRINTF(E_LOG, L_XCODE, "Can't proceed, codectype is unknown (null)\n"); DPRINTF(E_LOG, L_XCODE, "Can't determine transcode status, codec type is unknown\n");
return -1; return -1;
} }