From: Nicolas Frattaroli <ffmpeg@fratti.ch> To: ffmpeg-devel@ffmpeg.org Cc: "Ronald S . Bultje" <rsbultje@gmail.com>, Nicolas Frattaroli <ffmpeg@fratti.ch>, Lynne <dev@lynne.ee> Subject: [FFmpeg-devel] [PATCH v2] avformat/http: Add max_requested_size option Date: Sat, 29 Mar 2025 23:28:10 +0100 Message-ID: <20250329222809.606230-2-ffmpeg@fratti.ch> (raw) In some cases, webservers may return content more quickly if one sends many small requests rather than one big request. Clients may wish to opt-in to this workaround. For this purpose, add a max_requested_size AVOption, which returns EOF if max_requested_size is exceeded, causing a new request with an offset to be issued. In particular, this hack is useful for mpv, where it can be used to bypass YouTube's throttling. Signed-off-by: Nicolas Frattaroli <ffmpeg@fratti.ch> --- Changes in v2: - Renamed option from max_request_size to max_requested_size --- libavformat/http.c | 43 +++++++++++++++++++++++++++++++------------ 1 file changed, 31 insertions(+), 12 deletions(-) diff --git a/libavformat/http.c b/libavformat/http.c index 65ea5d993c..bcb4e0f4c4 100644 --- a/libavformat/http.c +++ b/libavformat/http.c @@ -77,7 +77,7 @@ typedef struct HTTPContext { /* Used if "Transfer-Encoding: chunked" otherwise -1. */ uint64_t chunksize; int chunkend; - uint64_t off, end_off, filesize; + uint64_t off, end_off, filesize, max_requested_size, range_end; char *uri; char *location; HTTPAuthState auth_state; @@ -176,6 +176,7 @@ static const AVOption options[] = { { "location", "The actual location of the data received", OFFSET(location), AV_OPT_TYPE_STRING, { .str = NULL }, 0, 0, D | E }, { "offset", "initial byte offset", OFFSET(off), AV_OPT_TYPE_INT64, { .i64 = 0 }, 0, INT64_MAX, D }, { "end_offset", "try to limit the request to bytes preceding this offset", OFFSET(end_off), AV_OPT_TYPE_INT64, { .i64 = 0 }, 0, INT64_MAX, D }, + { "max_requested_size", "try to limit the request to this many bytes, then reconnect", OFFSET(max_requested_size), AV_OPT_TYPE_INT64, { .i64 = 0 }, 0, INT64_MAX, D }, { "method", "Override the HTTP method or set the expected HTTP method from a client", OFFSET(method), AV_OPT_TYPE_STRING, { .str = NULL }, 0, 0, D | E }, { "reconnect", "auto reconnect after disconnect before EOF", OFFSET(reconnect), AV_OPT_TYPE_BOOL, { .i64 = 0 }, 0, 1, D }, { "reconnect_at_eof", "auto reconnect at EOF", OFFSET(reconnect_at_eof), AV_OPT_TYPE_BOOL, { .i64 = 0 }, 0, 1, D }, @@ -1524,10 +1525,16 @@ static int http_connect(URLContext *h, const char *path, const char *local_path, // Note: we send the Range header on purpose, even when we're probing, // since it allows us to detect more reliably if a (non-conforming) // server supports seeking by analysing the reply headers. - if (!has_header(s->headers, "\r\nRange: ") && !post && (s->off > 0 || s->end_off || s->seekable != 0)) { + if (!has_header(s->headers, "\r\nRange: ") && !post && (s->off > 0 || s->end_off || s->seekable != 0 || s->max_requested_size)) { av_bprintf(&request, "Range: bytes=%"PRIu64"-", s->off); - if (s->end_off) - av_bprintf(&request, "%"PRId64, s->end_off - 1); + if (s->end_off || s->max_requested_size) { + if (s->end_off && s->max_requested_size) { + s->range_end = FFMIN(s->end_off, s->off + s->max_requested_size); + } else { + s->range_end = FFMAX(s->end_off, s->off + s->max_requested_size); + } + av_bprintf(&request, "%"PRId64, s->range_end - 1); + } av_bprintf(&request, "\r\n"); } if (send_expect_100 && !has_header(s->headers, "\r\nExpect: ")) @@ -1666,7 +1673,16 @@ static int http_buf_read(URLContext *h, uint8_t *buf, int size) memcpy(buf, s->buf_ptr, len); s->buf_ptr += len; } else { - uint64_t target_end = s->end_off ? s->end_off : s->filesize; + uint64_t target_end = s->filesize; + if (s->end_off || s->max_requested_size) { + if (s->end_off && s->max_requested_size) { + target_end = FFMIN(s->end_off, s->range_end); + } else { + target_end = FFMAX(s->end_off, s->range_end); + } + target_end = FFMIN(target_end, s->filesize); + } + if ((!s->willclose || s->chunksize == UINT64_MAX) && s->off >= target_end) return AVERROR_EOF; len = ffurl_read(s->hd, buf, size); @@ -1769,13 +1785,16 @@ static int http_read_stream(URLContext *h, uint8_t *buf, int size) reconnect_delay_total > s->reconnect_delay_total_max) return AVERROR(EIO); - av_log(h, AV_LOG_WARNING, "Will reconnect at %"PRIu64" in %d second(s), error=%s.\n", s->off, reconnect_delay, av_err2str(read_ret)); - err = ff_network_sleep_interruptible(1000U*1000*reconnect_delay, &h->interrupt_callback); - if (err != AVERROR(ETIMEDOUT)) - return err; - reconnect_delay_total += reconnect_delay; - reconnect_delay = 1 + 2*reconnect_delay; - conn_attempts++; + if (!(s->max_requested_size > 0 && read_ret == AVERROR_EOF)) { + av_log(h, AV_LOG_WARNING, "Will reconnect at %"PRIu64" in %d second(s), error=%s.\n", + s->off, reconnect_delay, av_err2str(read_ret)); + err = ff_network_sleep_interruptible(1000U*1000*reconnect_delay, &h->interrupt_callback); + if (err != AVERROR(ETIMEDOUT)) + return err; + reconnect_delay_total += reconnect_delay; + reconnect_delay = 1 + 2*reconnect_delay; + conn_attempts++; + } seek_ret = http_seek_internal(h, target, SEEK_SET, 1); if (seek_ret >= 0 && seek_ret != target) { av_log(h, AV_LOG_ERROR, "Failed to reconnect at %"PRIu64".\n", target); -- 2.49.0 _______________________________________________ ffmpeg-devel mailing list ffmpeg-devel@ffmpeg.org https://ffmpeg.org/mailman/listinfo/ffmpeg-devel To unsubscribe, visit link above, or email ffmpeg-devel-request@ffmpeg.org with subject "unsubscribe".
reply other threads:[~2025-03-29 22:30 UTC|newest] Thread overview: [no followups] expand[flat|nested] mbox.gz Atom feed
Reply instructions: You may reply publicly to this message via plain-text email using any one of the following methods: * Save the following mbox file, import it into your mail client, and reply-to-all from there: mbox Avoid top-posting and favor interleaved quoting: https://en.wikipedia.org/wiki/Posting_style#Interleaved_style * Reply using the --to, --cc, and --in-reply-to switches of git-send-email(1): git send-email \ --in-reply-to=20250329222809.606230-2-ffmpeg@fratti.ch \ --to=ffmpeg@fratti.ch \ --cc=dev@lynne.ee \ --cc=ffmpeg-devel@ffmpeg.org \ --cc=rsbultje@gmail.com \ /path/to/YOUR_REPLY https://kernel.org/pub/software/scm/git/docs/git-send-email.html * If your mail client supports setting the In-Reply-To header via mailto: links, try the mailto: link
Git Inbox Mirror of the ffmpeg-devel mailing list - see https://ffmpeg.org/mailman/listinfo/ffmpeg-devel This inbox may be cloned and mirrored by anyone: git clone --mirror https://master.gitmailbox.com/ffmpegdev/0 ffmpegdev/git/0.git # If you have public-inbox 1.1+ installed, you may # initialize and index your mirror using the following commands: public-inbox-init -V2 ffmpegdev ffmpegdev/ https://master.gitmailbox.com/ffmpegdev \ ffmpegdev@gitmailbox.com public-inbox-index ffmpegdev Example config snippet for mirrors. AGPL code for this site: git clone https://public-inbox.org/public-inbox.git