/* * Copyright (C) 2016 Espen Jürgensen * * 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 #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef HAVE_LIBCURL #include #endif #include "http.h" #include "httpd.h" #include "logger.h" #include "misc.h" #include "conffile.h" /* Formats we can read so far */ #define PLAYLIST_UNK 0 #define PLAYLIST_PLS 1 #define PLAYLIST_M3U 2 /* ======================= libevent HTTP client =============================*/ // Number of seconds the client will wait for a response before aborting #define HTTP_CLIENT_TIMEOUT 8 /* The strict libevent api does not permit walking through an evkeyvalq and saving * all the http headers, so we predefine what we are looking for. You can add * extra headers here that you would like to save. */ static char *header_list[] = { "icy-name", "icy-description", "icy-metaint", "icy-genre", "Content-Type", }; /* Copies headers we are searching for from one keyval struct to another * */ static void headers_save(struct keyval *kv, struct evkeyvalq *headers) { const char *value; uint8_t *utf; int i; if (!kv || !headers) return; for (i = 0; i < (sizeof(header_list) / sizeof(header_list[0])); i++) if ( (value = evhttp_find_header(headers, header_list[i])) ) { utf = u8_strconv_from_encoding(value, "ISO−8859−1", iconveh_question_mark); if (!utf) continue; keyval_add(kv, header_list[i], (char *)utf); free(utf); } } static void request_cb(struct evhttp_request *req, void *arg) { struct http_client_ctx *ctx; const char *response_code_line; ctx = (struct http_client_ctx *)arg; if (ctx->headers_only) { ctx->ret = 0; event_base_loopbreak(ctx->evbase); return; } if (!req) { DPRINTF(E_WARN, L_HTTP, "Connection to %s failed: Connection timed out\n", ctx->url); goto connection_error; } ctx->response_code = evhttp_request_get_response_code(req); #ifndef HAVE_LIBEVENT2_OLD response_code_line = evhttp_request_get_response_code_line(req); #else response_code_line = "no error text"; #endif if (ctx->response_code == 0) { DPRINTF(E_WARN, L_HTTP, "Connection to %s failed: Connection refused\n", ctx->url); goto connection_error; } else if (ctx->response_code != 200) { DPRINTF(E_WARN, L_HTTP, "Connection to %s failed: %s (error %d)\n", ctx->url, response_code_line, ctx->response_code); goto connection_error; } ctx->ret = 0; if (ctx->input_headers) headers_save(ctx->input_headers, evhttp_request_get_input_headers(req)); if (ctx->input_body) evbuffer_add_buffer(ctx->input_body, evhttp_request_get_input_buffer(req)); event_base_loopbreak(ctx->evbase); return; connection_error: ctx->ret = -1; event_base_loopbreak(ctx->evbase); return; } /* This callback is only invoked if ctx->headers_only is set. Since that means * we only want headers, it will always return -1 to make evhttp close the * connection. The headers will be saved in a keyval struct in ctx, since we * cannot address the *evkeyvalq after the connection is free'd. */ #ifndef HAVE_LIBEVENT2_OLD static int request_header_cb(struct evhttp_request *req, void *arg) { struct http_client_ctx *ctx; ctx = (struct http_client_ctx *)arg; if (!ctx->input_headers) { DPRINTF(E_LOG, L_HTTP, "BUG: Header callback invoked but caller did not say where to save the headers\n"); return -1; } headers_save(ctx->input_headers, evhttp_request_get_input_headers(req)); return -1; } #endif static int http_client_request_impl(struct http_client_ctx *ctx) { struct evhttp_connection *evcon; struct evhttp_request *req; struct evkeyvalq *headers; struct evbuffer *output_buffer; struct onekeyval *okv; enum evhttp_cmd_type method; const char *user_agent; char host[512]; char host_port[1024]; char path[2048]; char tmp[128]; int port; int ret; ctx->ret = -1; av_url_split(NULL, 0, NULL, 0, host, sizeof(host), &port, path, sizeof(path), ctx->url); if (strlen(host) == 0) { DPRINTF(E_LOG, L_HTTP, "Error extracting hostname from URL: %s\n", ctx->url); return ctx->ret; } if (port <= 0) snprintf(host_port, sizeof(host_port), "%s", host); else snprintf(host_port, sizeof(host_port), "%s:%d", host, port); if (port <= 0) port = 80; if (strlen(path) == 0) { path[0] = '/'; path[1] = '\0'; } ctx->evbase = event_base_new(); if (!ctx->evbase) { DPRINTF(E_LOG, L_HTTP, "Could not create or find http client event base\n"); return ctx->ret; } evcon = evhttp_connection_base_new(ctx->evbase, NULL, host, (unsigned short)port); if (!evcon) { DPRINTF(E_LOG, L_HTTP, "Could not create connection to %s\n", host_port); event_base_free(ctx->evbase); return ctx->ret; } evhttp_connection_set_timeout(evcon, HTTP_CLIENT_TIMEOUT); /* Set up request */ req = evhttp_request_new(request_cb, ctx); if (!req) { DPRINTF(E_LOG, L_HTTP, "Could not create request to %s\n", host_port); evhttp_connection_free(evcon); event_base_free(ctx->evbase); return ctx->ret; } #ifndef HAVE_LIBEVENT2_OLD if (ctx->headers_only) evhttp_request_set_header_cb(req, request_header_cb); #endif headers = evhttp_request_get_output_headers(req); evhttp_add_header(headers, "Host", host_port); user_agent = cfg_getstr(cfg_getsec(cfg, "general"), "user_agent"); evhttp_add_header(headers, "User-Agent", user_agent); evhttp_add_header(headers, "Icy-MetaData", "1"); if (ctx->output_headers) { for (okv = ctx->output_headers->head; okv; okv = okv->next) evhttp_add_header(headers, okv->name, okv->value); } if (ctx->output_body) { output_buffer = evhttp_request_get_output_buffer(req); evbuffer_add(output_buffer, ctx->output_body, strlen(ctx->output_body)); evbuffer_add_printf(output_buffer, "\n"); snprintf(tmp, sizeof(tmp), "%zu", evbuffer_get_length(output_buffer)); evhttp_add_header(headers, "Content-Length", tmp); method = EVHTTP_REQ_POST; } else { evhttp_add_header(headers, "Content-Length", "0"); method = EVHTTP_REQ_GET; } /* Make request */ DPRINTF(E_INFO, L_HTTP, "Making %s request for http://%s%s\n", ((method==EVHTTP_REQ_GET) ? "GET" : "POST"), host_port, path); ret = evhttp_make_request(evcon, req, method, path); if (ret < 0) { DPRINTF(E_LOG, L_HTTP, "Error making request for http://%s%s\n", host_port, path); evhttp_connection_free(evcon); event_base_free(ctx->evbase); return ctx->ret; } event_base_dispatch(ctx->evbase); evhttp_connection_free(evcon); event_base_free(ctx->evbase); return ctx->ret; } #ifdef HAVE_LIBCURL static void curl_headers_save(struct keyval *kv, CURL *curl) { char *content_type; int ret; if (!kv || !curl) return; ret = curl_easy_getinfo(curl, CURLINFO_CONTENT_TYPE, &content_type); if (ret == CURLE_OK && content_type) { keyval_add(kv, "Content-Type", content_type); } } static size_t curl_request_cb(char *ptr, size_t size, size_t nmemb, void *userdata) { size_t realsize; struct http_client_ctx *ctx; int ret; realsize = size * nmemb; ctx = (struct http_client_ctx *)userdata; if (!ctx->input_body) return realsize; ret = evbuffer_add(ctx->input_body, ptr, realsize); if (ret < 0) { DPRINTF(E_LOG, L_HTTP, "Error adding reply from %s to input buffer\n", ctx->url); return 0; } return realsize; } static int https_client_request_impl(struct http_client_ctx *ctx) { CURL *curl; CURLcode res; struct curl_slist *headers; struct onekeyval *okv; const char *user_agent; char header[1024]; long response_code; curl = curl_easy_init(); if (!curl) { DPRINTF(E_LOG, L_HTTP, "Error: Could not get curl handle\n"); return -1; } user_agent = cfg_getstr(cfg_getsec(cfg, "general"), "user_agent"); curl_easy_setopt(curl, CURLOPT_USERAGENT, user_agent); curl_easy_setopt(curl, CURLOPT_URL, ctx->url); headers = NULL; if (ctx->output_headers) { for (okv = ctx->output_headers->head; okv; okv = okv->next) { snprintf(header, sizeof(header), "%s: %s", okv->name, okv->value); headers = curl_slist_append(headers, header); } curl_easy_setopt(curl, CURLOPT_HTTPHEADER, headers); } if (ctx->output_body) curl_easy_setopt(curl, CURLOPT_POSTFIELDS, ctx->output_body); curl_easy_setopt(curl, CURLOPT_TIMEOUT, HTTP_CLIENT_TIMEOUT); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, curl_request_cb); curl_easy_setopt(curl, CURLOPT_WRITEDATA, ctx); // Artwork and playlist requests might require redirects curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1); curl_easy_setopt(curl, CURLOPT_MAXREDIRS, 5); /* Make request */ DPRINTF(E_INFO, L_HTTP, "Making request for %s\n", ctx->url); res = curl_easy_perform(curl); if (res != CURLE_OK) { DPRINTF(E_LOG, L_HTTP, "Request to %s failed: %s\n", ctx->url, curl_easy_strerror(res)); curl_slist_free_all(headers); curl_easy_cleanup(curl); return -1; } curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &response_code); ctx->response_code = (int) response_code; curl_headers_save(ctx->input_headers, curl); curl_slist_free_all(headers); curl_easy_cleanup(curl); return 0; } #endif /* HAVE_LIBCURL */ int http_client_request(struct http_client_ctx *ctx) { if (strncmp(ctx->url, "http:", strlen("http:")) == 0) return http_client_request_impl(ctx); #ifdef HAVE_LIBCURL if (strncmp(ctx->url, "https:", strlen("https:")) == 0) return https_client_request_impl(ctx); #endif DPRINTF(E_LOG, L_HTTP, "Request for %s is not supported (not built with libcurl?)\n", ctx->url); return -1; } char * http_form_urlencode(struct keyval *kv) { struct evbuffer *evbuf; struct onekeyval *okv; char *body; char *k; char *v; evbuf = evbuffer_new(); for (okv = kv->head; okv; okv = okv->next) { k = evhttp_encode_uri(okv->name); if (!k) continue; v = evhttp_encode_uri(okv->value); if (!v) { free(k); continue; } evbuffer_add_printf(evbuf, "%s=%s", k, v); if (okv->next) evbuffer_add_printf(evbuf, "&"); free(k); free(v); } evbuffer_add(evbuf, "\n", 1); body = evbuffer_readln(evbuf, NULL, EVBUFFER_EOL_ANY); evbuffer_free(evbuf); DPRINTF(E_DBG, L_HTTP, "Parameters in request are: %s\n", body); return body; } int http_stream_setup(char **stream, const char *url) { struct http_client_ctx ctx; struct httpd_uri_parsed *parsed; struct evbuffer *evbuf; const char *ext; char *line; char *pos; int ret; int n; int pl_format; bool in_playlist; *stream = NULL; parsed = httpd_uri_parse(url); if (!parsed) { DPRINTF(E_LOG, L_HTTP, "Couldn't parse internet playlist: '%s'\n", url); return -1; } // parsed->path does not include query or fragment, so should work with any url's // e.g. http://yp.shoutcast.com/sbin/tunein-station.pls?id=99179772#Air Jazz pl_format = PLAYLIST_UNK; if (parsed->path && (ext = strrchr(parsed->path, '.'))) { if (strcasecmp(ext, ".m3u") == 0) pl_format = PLAYLIST_M3U; else if (strcasecmp(ext, ".pls") == 0) pl_format = PLAYLIST_PLS; } httpd_uri_free(parsed); if (pl_format==PLAYLIST_UNK) { *stream = strdup(url); return 0; } // It was a m3u or pls playlist, so now retrieve it memset(&ctx, 0, sizeof(struct http_client_ctx)); evbuf = evbuffer_new(); if (!evbuf) return -1; ctx.url = url; ctx.input_body = evbuf; ret = http_client_request(&ctx); if (ret < 0) { DPRINTF(E_LOG, L_HTTP, "Couldn't fetch internet playlist: %s\n", url); evbuffer_free(evbuf); return -1; } // Pad with CRLF because evbuffer_readln() might not read the last line otherwise evbuffer_add(ctx.input_body, "\r\n", 2); /* Read the playlist until the first stream link is found, but give up if * nothing is found in the first 10 lines */ in_playlist = false; n = 0; while ((line = evbuffer_readln(ctx.input_body, NULL, EVBUFFER_EOL_ANY)) && (n < 10)) { // Skip comments and blank lines without counting for the limit if (pl_format == PLAYLIST_M3U && (line[0] == '#' || line[0] == '\0')) goto line_done; n++; if (pl_format == PLAYLIST_PLS && !in_playlist) { if (strncasecmp(line, "[playlist]", strlen("[playlist]")) == 0) { in_playlist = true; n = 0; } goto line_done; } if (pl_format == PLAYLIST_PLS) { pos = line; while (*pos == ' ') ++pos; // We are only interested in `FileN=http://foo/bar.mp3` entries if (strncasecmp(pos, "file", strlen("file")) != 0) goto line_done; while (*pos != '=' && *pos != '\0') ++pos; if (*pos == '\0') goto line_done; ++pos; while (*pos == ' ') ++pos; // allocate the value part and proceed as with m3u pos = strdup(pos); free(line); line = pos; } if (strncasecmp(line, "http://", strlen("http://")) == 0) { DPRINTF(E_DBG, L_HTTP, "Found internet playlist stream (line %d): %s\n", n, line); n = -1; break; } line_done: free(line); } evbuffer_free(ctx.input_body); if (n != -1) { DPRINTF(E_LOG, L_HTTP, "Couldn't find stream in internet playlist: %s\n", url); return -1; } *stream = line; return 0; } /* ======================= ICY metadata handling =============================*/ #if LIBAVFORMAT_VERSION_MAJOR >= 56 || (LIBAVFORMAT_VERSION_MAJOR == 55 && LIBAVFORMAT_VERSION_MINOR >= 13) static int metadata_packet_get(struct http_icy_metadata *metadata, AVFormatContext *fmtctx) { uint8_t *buffer; char *icy_token; char *save_pr; char *ptr; char *end; av_opt_get(fmtctx, "icy_metadata_packet", AV_OPT_SEARCH_CHILDREN, &buffer); if (!buffer) return -1; icy_token = strtok_r((char *)buffer, ";", &save_pr); while (icy_token != NULL) { ptr = strchr(icy_token, '='); if (!ptr || (ptr[1] == '\0')) { icy_token = strtok_r(NULL, ";", &save_pr); continue; } ptr++; if (ptr[0] == '\'') ptr++; end = strrchr(ptr, '\''); if (end) *end = '\0'; if ((strncmp(icy_token, "StreamTitle", strlen("StreamTitle")) == 0) && !metadata->title) { metadata->title = ptr; /* Dash separates artist from title, if no dash assume all is title */ ptr = strstr(ptr, " - "); if (ptr) { *ptr = '\0'; metadata->artist = strdup(metadata->title); *ptr = ' '; metadata->title = strdup(ptr + 3); } else if (strlen(metadata->title) == 0) metadata->title = NULL; else metadata->title = strdup(metadata->title); } else if ((strncmp(icy_token, "StreamUrl", strlen("StreamUrl")) == 0) && !metadata->url && strlen(ptr) > 0) { metadata->url = strdup(ptr); } if (end) *end = '\''; icy_token = strtok_r(NULL, ";", &save_pr); } av_free(buffer); if (metadata->title) metadata->hash = djb_hash(metadata->title, strlen(metadata->title)); return 0; } static int metadata_header_get(struct http_icy_metadata *metadata, AVFormatContext *fmtctx) { uint8_t *buffer; uint8_t *utf; char *icy_token; char *save_pr; char *ptr; av_opt_get(fmtctx, "icy_metadata_headers", AV_OPT_SEARCH_CHILDREN, &buffer); if (!buffer) return -1; /* Headers are ascii or iso-8859-1 according to: * http://www.w3.org/Protocols/rfc2616/rfc2616-sec2.html#sec2.2 */ utf = u8_strconv_from_encoding((char *)buffer, "ISO−8859−1", iconveh_question_mark); av_free(buffer); if (!utf) return -1; icy_token = strtok_r((char *)utf, "\r\n", &save_pr); while (icy_token != NULL) { ptr = strchr(icy_token, ':'); if (!ptr || (ptr[1] == '\0')) { icy_token = strtok_r(NULL, "\r\n", &save_pr); continue; } ptr++; if (ptr[0] == ' ') ptr++; if ((strncmp(icy_token, "icy-name", strlen("icy-name")) == 0) && !metadata->name) metadata->name = strdup(ptr); else if ((strncmp(icy_token, "icy-description", strlen("icy-description")) == 0) && !metadata->description) metadata->description = strdup(ptr); else if ((strncmp(icy_token, "icy-genre", strlen("icy-genre")) == 0) && !metadata->genre) metadata->genre = strdup(ptr); icy_token = strtok_r(NULL, "\r\n", &save_pr); } free(utf); return 0; } struct http_icy_metadata * http_icy_metadata_get(AVFormatContext *fmtctx, int packet_only) { struct http_icy_metadata *metadata; int got_packet; int got_header; CHECK_NULL(L_HTTP, metadata = calloc(1, sizeof(struct http_icy_metadata))); got_packet = (metadata_packet_get(metadata, fmtctx) == 0); got_header = (!packet_only) && (metadata_header_get(metadata, fmtctx) == 0); if (!got_packet && !got_header) { free(metadata); return NULL; } /* DPRINTF(E_DBG, L_HTTP, "Found ICY: N %s, D %s, G %s, T %s, A %s, U %s, I %" PRIu32 "\n", metadata->name, metadata->description, metadata->genre, metadata->title, metadata->artist, metadata->url, metadata->hash ); */ return metadata; } #elif defined(HAVE_LIBEVENT2_OLD) struct http_icy_metadata * http_icy_metadata_get(AVFormatContext *fmtctx, int packet_only) { DPRINTF(E_INFO, L_HTTP, "Skipping Shoutcast metadata request for %s (requires libevent>=2.1.4 or libav 10)\n", fmtctx->filename); return NULL; } #else /* Earlier versions of ffmpeg/libav do not seem to allow access to the http * headers, so we must instead open the stream ourselves to get the metadata. * Sorry about the extra connections, you radio streaming people! * * It is not possible to get the packet metadata with these versions of ffmpeg */ struct http_icy_metadata * http_icy_metadata_get(AVFormatContext *fmtctx, int packet_only) { struct http_icy_metadata *metadata; struct http_client_ctx ctx; struct keyval *kv; const char *value; int got_header; int ret; /* Can only get header metadata */ if (packet_only) return NULL; kv = keyval_alloc(); if (!kv) return NULL; memset(&ctx, 0, sizeof(struct http_client_ctx)); ctx.url = fmtctx->filename; ctx.input_headers = kv; ctx.headers_only = 1; ctx.input_body = NULL; ret = http_client_request(&ctx); if (ret < 0) { DPRINTF(E_LOG, L_HTTP, "Error fetching %s\n", fmtctx->filename); free(kv); return NULL; } CHECK_NULL(L_HTTP, metadata = calloc(1, sizeof(struct http_icy_metadata))); got_header = 0; if ( (value = keyval_get(ctx.input_headers, "icy-name")) ) { metadata->name = strdup(value); got_header = 1; } if ( (value = keyval_get(ctx.input_headers, "icy-description")) ) { metadata->description = strdup(value); got_header = 1; } if ( (value = keyval_get(ctx.input_headers, "icy-genre")) ) { metadata->genre = strdup(value); got_header = 1; } keyval_clear(kv); free(kv); if (!got_header) { free(metadata); return NULL; } /* DPRINTF(E_DBG, L_HTTP, "Found ICY: N %s, D %s, G %s, T %s, A %s, U %s, I %" PRIu32 "\n", metadata->name, metadata->description, metadata->genre, metadata->title, metadata->artist, metadata->url, metadata->hash );*/ return metadata; } #endif void http_icy_metadata_free(struct http_icy_metadata *metadata, int content_only) { if (!metadata) return; free(metadata->name); free(metadata->description); free(metadata->genre); free(metadata->title); free(metadata->artist); free(metadata->url); free(metadata); }