Commit d24e08e9 authored by Karthick J's avatar Karthick J Committed by Michael Niedermayer

avformat/dashenc: Added configuration to override HTTP User-Agent

Reviewed-by: 's avatarSteven Liu <lingjiujianke@gmail.com>
Signed-off-by: 's avatarMichael Niedermayer <michael@niedermayer.cc>
parent 2d9cf3bf
...@@ -247,6 +247,8 @@ DASH-templated name to used for the initialization segment. Default is "init-str ...@@ -247,6 +247,8 @@ DASH-templated name to used for the initialization segment. Default is "init-str
DASH-templated name to used for the media segments. Default is "chunk-stream$RepresentationID$-$Number%05d$.m4s" DASH-templated name to used for the media segments. Default is "chunk-stream$RepresentationID$-$Number%05d$.m4s"
@item -utc_timing_url @var{utc_url} @item -utc_timing_url @var{utc_url}
URL of the page that will return the UTC timestamp in ISO format. Example: "https://time.akamai.com/?iso" URL of the page that will return the UTC timestamp in ISO format. Example: "https://time.akamai.com/?iso"
@item -http_user_agent @var{user_agent}
Override User-Agent field in HTTP header. Applicable only for HTTP output.
@item -adaptation_sets @var{adaptation_sets} @item -adaptation_sets @var{adaptation_sets}
Assign streams to AdaptationSets. Syntax is "id=x,streams=a,b,c id=y,streams=d,e" with x and y being the IDs Assign streams to AdaptationSets. Syntax is "id=x,streams=a,b,c id=y,streams=d,e" with x and y being the IDs
of the adaptation sets and a,b,c,d and e are the indices of the mapped streams. of the adaptation sets and a,b,c,d and e are the indices of the mapped streams.
......
...@@ -100,6 +100,7 @@ typedef struct DASHContext { ...@@ -100,6 +100,7 @@ typedef struct DASHContext {
const char *init_seg_name; const char *init_seg_name;
const char *media_seg_name; const char *media_seg_name;
const char *utc_timing_url; const char *utc_timing_url;
const char *user_agent;
} DASHContext; } DASHContext;
static struct codec_string { static struct codec_string {
...@@ -210,6 +211,12 @@ static int flush_dynbuf(OutputStream *os, int *range_length) ...@@ -210,6 +211,12 @@ static int flush_dynbuf(OutputStream *os, int *range_length)
return avio_open_dyn_buf(&os->ctx->pb); return avio_open_dyn_buf(&os->ctx->pb);
} }
static void set_http_options(AVDictionary **options, DASHContext *c)
{
if (c->user_agent)
av_dict_set(options, "user_agent", c->user_agent, 0);
}
static int flush_init_segment(AVFormatContext *s, OutputStream *os) static int flush_init_segment(AVFormatContext *s, OutputStream *os)
{ {
DASHContext *c = s->priv_data; DASHContext *c = s->priv_data;
...@@ -575,16 +582,19 @@ static int write_manifest(AVFormatContext *s, int final) ...@@ -575,16 +582,19 @@ static int write_manifest(AVFormatContext *s, int final)
int use_rename = proto && !strcmp(proto, "file"); int use_rename = proto && !strcmp(proto, "file");
static unsigned int warned_non_file = 0; static unsigned int warned_non_file = 0;
AVDictionaryEntry *title = av_dict_get(s->metadata, "title", NULL, 0); AVDictionaryEntry *title = av_dict_get(s->metadata, "title", NULL, 0);
AVDictionary *opts = NULL;
if (!use_rename && !warned_non_file++) if (!use_rename && !warned_non_file++)
av_log(s, AV_LOG_ERROR, "Cannot use rename on non file protocol, this may lead to races and temporary partial files\n"); av_log(s, AV_LOG_ERROR, "Cannot use rename on non file protocol, this may lead to races and temporary partial files\n");
snprintf(temp_filename, sizeof(temp_filename), use_rename ? "%s.tmp" : "%s", s->filename); snprintf(temp_filename, sizeof(temp_filename), use_rename ? "%s.tmp" : "%s", s->filename);
ret = s->io_open(s, &out, temp_filename, AVIO_FLAG_WRITE, NULL); set_http_options(&opts, c);
ret = s->io_open(s, &out, temp_filename, AVIO_FLAG_WRITE, &opts);
if (ret < 0) { if (ret < 0) {
av_log(s, AV_LOG_ERROR, "Unable to open %s for writing\n", temp_filename); av_log(s, AV_LOG_ERROR, "Unable to open %s for writing\n", temp_filename);
return ret; return ret;
} }
av_dict_free(&opts);
avio_printf(out, "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n"); avio_printf(out, "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n");
avio_printf(out, "<MPD xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\n" avio_printf(out, "<MPD xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\n"
"\txmlns=\"urn:mpeg:dash:schema:mpd:2011\"\n" "\txmlns=\"urn:mpeg:dash:schema:mpd:2011\"\n"
...@@ -768,9 +778,11 @@ static int dash_init(AVFormatContext *s) ...@@ -768,9 +778,11 @@ static int dash_init(AVFormatContext *s)
ff_dash_fill_tmpl_params(os->initfile, sizeof(os->initfile), c->init_seg_name, i, 0, os->bit_rate, 0); ff_dash_fill_tmpl_params(os->initfile, sizeof(os->initfile), c->init_seg_name, i, 0, os->bit_rate, 0);
} }
snprintf(filename, sizeof(filename), "%s%s", c->dirname, os->initfile); snprintf(filename, sizeof(filename), "%s%s", c->dirname, os->initfile);
ret = s->io_open(s, &os->out, filename, AVIO_FLAG_WRITE, NULL); set_http_options(&opts, c);
ret = s->io_open(s, &os->out, filename, AVIO_FLAG_WRITE, &opts);
if (ret < 0) if (ret < 0)
return ret; return ret;
av_dict_free(&opts);
os->init_start_pos = 0; os->init_start_pos = 0;
if (!strcmp(os->format_name, "mp4")) { if (!strcmp(os->format_name, "mp4")) {
...@@ -974,12 +986,15 @@ static int dash_flush(AVFormatContext *s, int final, int stream) ...@@ -974,12 +986,15 @@ static int dash_flush(AVFormatContext *s, int final, int stream)
} }
if (!c->single_file) { if (!c->single_file) {
AVDictionary *opts = NULL;
ff_dash_fill_tmpl_params(filename, sizeof(filename), c->media_seg_name, i, os->segment_index, os->bit_rate, os->start_pts); ff_dash_fill_tmpl_params(filename, sizeof(filename), c->media_seg_name, i, os->segment_index, os->bit_rate, os->start_pts);
snprintf(full_path, sizeof(full_path), "%s%s", c->dirname, filename); snprintf(full_path, sizeof(full_path), "%s%s", c->dirname, filename);
snprintf(temp_path, sizeof(temp_path), use_rename ? "%s.tmp" : "%s", full_path); snprintf(temp_path, sizeof(temp_path), use_rename ? "%s.tmp" : "%s", full_path);
ret = s->io_open(s, &os->out, temp_path, AVIO_FLAG_WRITE, NULL); set_http_options(&opts, c);
ret = s->io_open(s, &os->out, temp_path, AVIO_FLAG_WRITE, &opts);
if (ret < 0) if (ret < 0)
break; break;
av_dict_free(&opts);
if (!strcmp(os->format_name, "mp4")) if (!strcmp(os->format_name, "mp4"))
write_styp(os->ctx->pb); write_styp(os->ctx->pb);
} else { } else {
...@@ -1190,6 +1205,7 @@ static const AVOption options[] = { ...@@ -1190,6 +1205,7 @@ static const AVOption options[] = {
{ "init_seg_name", "DASH-templated name to used for the initialization segment", OFFSET(init_seg_name), AV_OPT_TYPE_STRING, {.str = "init-stream$RepresentationID$.m4s"}, 0, 0, E }, { "init_seg_name", "DASH-templated name to used for the initialization segment", OFFSET(init_seg_name), AV_OPT_TYPE_STRING, {.str = "init-stream$RepresentationID$.m4s"}, 0, 0, E },
{ "media_seg_name", "DASH-templated name to used for the media segments", OFFSET(media_seg_name), AV_OPT_TYPE_STRING, {.str = "chunk-stream$RepresentationID$-$Number%05d$.m4s"}, 0, 0, E }, { "media_seg_name", "DASH-templated name to used for the media segments", OFFSET(media_seg_name), AV_OPT_TYPE_STRING, {.str = "chunk-stream$RepresentationID$-$Number%05d$.m4s"}, 0, 0, E },
{ "utc_timing_url", "URL of the page that will return the UTC timestamp in ISO format", OFFSET(utc_timing_url), AV_OPT_TYPE_STRING, { 0 }, 0, 0, E }, { "utc_timing_url", "URL of the page that will return the UTC timestamp in ISO format", OFFSET(utc_timing_url), AV_OPT_TYPE_STRING, { 0 }, 0, 0, E },
{ "http_user_agent", "override User-Agent field in HTTP header", OFFSET(user_agent), AV_OPT_TYPE_STRING, {.str = NULL}, 0, 0, E},
{ NULL }, { NULL },
}; };
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment