On Sun, Jan 15 2023, Jeff King wrote: > The CURLOPT_PROTOCOLS (and matching CURLOPT_REDIR_PROTOCOLS) flag was > deprecated in curl 7.85.0, and using it generate compiler warnings as of > curl 7.87.0. The path forward is to use CURLOPT_PROTOCOLS_STR, but we > can't just do so unilaterally, as it was only introduced less than a > year ago in 7.85.0. > > Until that version becomes ubiquitous, we have to either disable the > deprecation warning or conditionally use the "STR" variant on newer > versions of libcurl. This patch switches to the new variant, which is > nice for two reasons: > > - we don't have to worry that silencing curl's deprecation warnings > might cause us to miss other more useful ones > > - we'd eventually want to move to the new variant anyway, so this gets > us set up (albeit with some extra ugly boilerplate for the > conditional) > > There are a lot of ways to split up the two cases. One way would be to > abstract the storage type (strbuf versus a long), how to append > (strbuf_addstr vs bitwise OR), how to initialize, which CURLOPT to use, > and so on. But the resulting code looks pretty magical: > > GIT_CURL_PROTOCOL_TYPE allowed = GIT_CURL_PROTOCOL_TYPE_INIT; > if (...http is allowed...) > GIT_CURL_PROTOCOL_APPEND(&allowed, "http", CURLOPT_HTTP); > > and you end up with more "#define GIT_CURL_PROTOCOL_TYPE" macros than > actual code. > > On the other end of the spectrum, we could just implement two separate > functions, one that handles a string list and one that handles bits. But > then we end up repeating our list of protocols (http, https, ftp, ftp). > > This patch takes the middle ground. The run-time code is always there to > handle both types, and we just choose which one to feed to curl. > > Signed-off-by: Jeff King <peff@xxxxxxxx> > --- > git-curl-compat.h | 8 ++++++++ > http.c | 41 ++++++++++++++++++++++++++++++++++------- > 2 files changed, 42 insertions(+), 7 deletions(-) > > diff --git a/git-curl-compat.h b/git-curl-compat.h > index 56a83b6bbd..fd96b3cdff 100644 > --- a/git-curl-compat.h > +++ b/git-curl-compat.h > @@ -126,4 +126,12 @@ > #define GIT_CURL_HAVE_CURLSSLSET_NO_BACKENDS > #endif > > +/** > + * CURLOPT_PROTOCOLS_STR and CURLOPT_REDIR_PROTOCOLS_STR were added in 7.85.0, > + * released in August 2022. > + */ > +#if LIBCURL_VERSION_NUM >= 0x075500 > +#define GIT_CURL_HAVE_CURLOPT_PROTOCOLS_STR 1 > +#endif > + > #endif Thanks, I hadn't had time to comment on this yet, but was wondering what this had to do with "CI" or "DEVEDEVELOPER_CFLAGS", the "CI" just seems to be "where we happened to spot this", and as has been noted this is also an issue without DEVELOPER_CFLAGS, we just happen to have -Werror there. But this is the right direction, and the reason we have git-curl-compat.h. > diff --git a/http.c b/http.c > index ca0fe80ddb..e529ebc993 100644 > --- a/http.c > +++ b/http.c > @@ -764,18 +764,29 @@ void setup_curl_trace(CURL *handle) > curl_easy_setopt(handle, CURLOPT_DEBUGDATA, NULL); > } > > -static long get_curl_allowed_protocols(int from_user) > +static void proto_list_append(struct strbuf *list_str, const char *proto_str, > + long *list_bits, long proto_bits) > +{ > + *list_bits |= proto_bits; > + if (list_str) { > + if (list_str->len) > + strbuf_addch(list_str, ','); > + strbuf_addstr(list_str, proto_str); > + } > +} Nit: It would be nice (especially in this even smaller function) to carry forward the name the parent get_curl_allowed_protocols() uses, i.e. just "list", not "list_str", ditto "proto" rather than "proto_str". > +#ifdef GIT_CURL_HAVE_CURLOPT_PROTOCOLS_STR > + { > + struct strbuf buf = STRBUF_INIT; > + > + get_curl_allowed_protocols(0, &buf); > + curl_easy_setopt(result, CURLOPT_REDIR_PROTOCOLS_STR, buf.buf); > + strbuf_reset(&buf); > + > + get_curl_allowed_protocols(-1, &buf); > + curl_easy_setopt(result, CURLOPT_PROTOCOLS_STR, buf.buf); > + strbuf_release(&buf); > + } > +#else > curl_easy_setopt(result, CURLOPT_REDIR_PROTOCOLS, > - get_curl_allowed_protocols(0)); > + get_curl_allowed_protocols(0, NULL)); > curl_easy_setopt(result, CURLOPT_PROTOCOLS, > - get_curl_allowed_protocols(-1)); > + get_curl_allowed_protocols(-1, NULL)); > +#endif > + > if (getenv("GIT_CURL_VERBOSE")) > http_trace_curl_no_data(); > setup_curl_trace(result); I wonder if it isn't better to avoid conditionally compiled code here if we can avoid it, i.e. just define GIT_* dummy fallbacks, and only use these if GIT_CURL_HAVE_CURLOPT_PROTOCOLS_STR is true. I.e. something like the below squashed in. diff --git a/git-curl-compat.h b/git-curl-compat.h index fd96b3cdffd..3bc6e151ca7 100644 --- a/git-curl-compat.h +++ b/git-curl-compat.h @@ -130,8 +130,13 @@ * CURLOPT_PROTOCOLS_STR and CURLOPT_REDIR_PROTOCOLS_STR were added in 7.85.0, * released in August 2022. */ -#if LIBCURL_VERSION_NUM >= 0x075500 -#define GIT_CURL_HAVE_CURLOPT_PROTOCOLS_STR 1 +#define GIT_CURL_HAVE_CURLOPT_PROTOCOLS_STR (LIBCURL_VERSION_NUM >= 0x075500) +#if GIT_CURL_HAVE_CURLOPT_PROTOCOLS_STR +#define GIT_CURLOPT_REDIR_PROTOCOLS_STR CURLOPT_REDIR_PROTOCOLS_STR +#define GIT_CURLOPT_PROTOCOLS_STR CURLOPT_PROTOCOLS_STR +#else +#define GIT_CURLOPT_REDIR_PROTOCOLS_STR 0 +#define GIT_CURLOPT_PROTOCOLS_STR 0 #endif #endif diff --git a/http.c b/http.c index e529ebc993d..3224e897f02 100644 --- a/http.c +++ b/http.c @@ -933,24 +933,22 @@ static CURL *get_curl_handle(void) curl_easy_setopt(result, CURLOPT_MAXREDIRS, 20); curl_easy_setopt(result, CURLOPT_POSTREDIR, CURL_REDIR_POST_ALL); -#ifdef GIT_CURL_HAVE_CURLOPT_PROTOCOLS_STR - { + if (GIT_CURL_HAVE_CURLOPT_PROTOCOLS_STR) { struct strbuf buf = STRBUF_INIT; get_curl_allowed_protocols(0, &buf); - curl_easy_setopt(result, CURLOPT_REDIR_PROTOCOLS_STR, buf.buf); + curl_easy_setopt(result, GIT_CURLOPT_REDIR_PROTOCOLS_STR, buf.buf); strbuf_reset(&buf); get_curl_allowed_protocols(-1, &buf); - curl_easy_setopt(result, CURLOPT_PROTOCOLS_STR, buf.buf); + curl_easy_setopt(result, GIT_CURLOPT_PROTOCOLS_STR, buf.buf); strbuf_release(&buf); + } else { + curl_easy_setopt(result, CURLOPT_REDIR_PROTOCOLS, + get_curl_allowed_protocols(0, NULL)); + curl_easy_setopt(result, CURLOPT_PROTOCOLS, + get_curl_allowed_protocols(-1, NULL)); } -#else - curl_easy_setopt(result, CURLOPT_REDIR_PROTOCOLS, - get_curl_allowed_protocols(0, NULL)); - curl_easy_setopt(result, CURLOPT_PROTOCOLS, - get_curl_allowed_protocols(-1, NULL)); -#endif if (getenv("GIT_CURL_VERBOSE")) http_trace_curl_no_data();