lib/fetch: cleanup happy eyeballs and add verbose logging
This commit is contained in:
parent
550bdfdcda
commit
81a736a5ac
@ -452,7 +452,7 @@ fetch_socks5(conn_t *conn, struct url *url, struct url *socks, int verbose)
|
|||||||
#define UNREACH_IPV6 0x01
|
#define UNREACH_IPV6 0x01
|
||||||
#define UNREACH_IPV4 0x10
|
#define UNREACH_IPV4 0x10
|
||||||
static int
|
static int
|
||||||
happy_eyeballs_connect(struct addrinfo *res0)
|
happy_eyeballs_connect(struct addrinfo *res0, int verbose)
|
||||||
{
|
{
|
||||||
static int unreach = 0;
|
static int unreach = 0;
|
||||||
struct pollfd *pfd;
|
struct pollfd *pfd;
|
||||||
@ -474,26 +474,25 @@ happy_eyeballs_connect(struct addrinfo *res0)
|
|||||||
case AF_INET: n4++; break;
|
case AF_INET: n4++; break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (n4+n6 == 0 || !(pfd = calloc(n4+n6, sizeof (struct pollfd))))
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
#ifdef FULL_DEBUG
|
#ifdef FULL_DEBUG
|
||||||
fetch_info("got %d A and %d AAAA records", n4, n6);
|
fetch_info("got %d A and %d AAAA records", n4, n6);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
i4 = i6 = 0;
|
i4 = i6 = 0;
|
||||||
if (getenv("FORCE_IPV4"))
|
if (unreach & UNREACH_IPV6 || getenv("FORCE_IPV4"))
|
||||||
i6 = n6;
|
i6 = n6;
|
||||||
if (getenv("FORCE_IPV6"))
|
if (unreach & UNREACH_IPV4 || getenv("FORCE_IPV6"))
|
||||||
i4 = n4;
|
i4 = n4;
|
||||||
|
|
||||||
if (unreach & UNREACH_IPV6)
|
if (n6+n4 == 0 || i6+i4 == n6+n4) {
|
||||||
i6 = n6;
|
netdb_seterr(EAI_FAIL);
|
||||||
if (unreach & UNREACH_IPV4)
|
return -1;
|
||||||
i4 = n4;
|
}
|
||||||
|
|
||||||
if (i6+i4 == n6+n4)
|
if (!(pfd = calloc(n4+n6, sizeof (struct pollfd)))) {
|
||||||
goto error;
|
fetch_syserr();
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
res = NULL;
|
res = NULL;
|
||||||
for (;;) {
|
for (;;) {
|
||||||
@ -502,8 +501,9 @@ happy_eyeballs_connect(struct addrinfo *res0)
|
|||||||
unsigned short family = 0;
|
unsigned short family = 0;
|
||||||
|
|
||||||
#ifdef FULL_DEBUG
|
#ifdef FULL_DEBUG
|
||||||
fetch_info("happy eyeballs state: i4=%u n4=%u i6=%u n6=%u"
|
if (verbose)
|
||||||
" attempts=%u waiting=%u", i4, n4, i6, n6, attempts, waiting);
|
fetch_info("happy eyeballs state: i4=%u n4=%u i6=%u n6=%u"
|
||||||
|
" attempts=%u waiting=%u", i4, n4, i6, n6, attempts, waiting);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (i6+i4 < n6+n4) {
|
if (i6+i4 < n6+n4) {
|
||||||
@ -523,9 +523,8 @@ happy_eyeballs_connect(struct addrinfo *res0)
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* no more connections to try */
|
/* no more connections to try */
|
||||||
#ifdef FULL_DEBUG
|
if (verbose)
|
||||||
fetch_info("attempted to connect to all addresses, waiting...");
|
fetch_info("attempted to connect to all addresses, waiting...");
|
||||||
#endif
|
|
||||||
timeout = fetchConnTimeout;
|
timeout = fetchConnTimeout;
|
||||||
done = 1;
|
done = 1;
|
||||||
goto wait;
|
goto wait;
|
||||||
@ -545,11 +544,13 @@ happy_eyeballs_connect(struct addrinfo *res0)
|
|||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (res == NULL)
|
if (res == NULL) {
|
||||||
goto error;
|
netdb_seterr(EAI_FAIL);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
if ((sd = socket(res->ai_family, res->ai_socktype | SOCK_NONBLOCK,
|
if ((sd = socket(res->ai_family, res->ai_socktype | SOCK_NONBLOCK,
|
||||||
res->ai_protocol)) == -1)
|
res->ai_protocol)) == -1)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (bindaddr != NULL && *bindaddr != '\0' &&
|
if (bindaddr != NULL && *bindaddr != '\0' &&
|
||||||
@ -559,11 +560,14 @@ happy_eyeballs_connect(struct addrinfo *res0)
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
if (verbose) {
|
||||||
char hbuf[1025];
|
char hbuf[1025];
|
||||||
if (getnameinfo(res->ai_addr, res->ai_addrlen, hbuf, sizeof(hbuf), NULL,
|
if (getnameinfo(res->ai_addr, res->ai_addrlen, hbuf, sizeof(hbuf), NULL,
|
||||||
0, NI_NUMERICHOST) == 0)
|
0, NI_NUMERICHOST) == 0)
|
||||||
fetch_info("connecting to %s", hbuf);
|
fetch_info("connecting to %s:%d", hbuf,
|
||||||
|
htons(res->ai_family == AF_INET
|
||||||
|
? ((struct sockaddr_in *)(res->ai_addr))->sin_port
|
||||||
|
: ((struct sockaddr_in6 *)(res->ai_addr))->sin6_port));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (connect(sd, res->ai_addr, res->ai_addrlen) == -1) {
|
if (connect(sd, res->ai_addr, res->ai_addrlen) == -1) {
|
||||||
@ -595,10 +599,9 @@ happy_eyeballs_connect(struct addrinfo *res0)
|
|||||||
waiting++;
|
waiting++;
|
||||||
wait:
|
wait:
|
||||||
if (!attempts) {
|
if (!attempts) {
|
||||||
error:
|
|
||||||
netdb_seterr(EAI_FAIL);
|
netdb_seterr(EAI_FAIL);
|
||||||
free(pfd);
|
rv = -1;
|
||||||
return -1;
|
goto out;
|
||||||
}
|
}
|
||||||
for (i = 0; i < attempts; i++) {
|
for (i = 0; i < attempts; i++) {
|
||||||
pfd[i].revents = pfd[i].events = 0;
|
pfd[i].revents = pfd[i].events = 0;
|
||||||
@ -637,8 +640,11 @@ error:
|
|||||||
}
|
}
|
||||||
if (!waiting)
|
if (!waiting)
|
||||||
break;
|
break;
|
||||||
|
else if (done)
|
||||||
|
goto wait;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
for (i = 0; i < attempts; i++)
|
for (i = 0; i < attempts; i++)
|
||||||
if ((rv == -1 || rv != pfd[i].fd) && pfd[i].fd != -1)
|
if ((rv == -1 || rv != pfd[i].fd) && pfd[i].fd != -1)
|
||||||
close(pfd[i].fd);
|
close(pfd[i].fd);
|
||||||
@ -703,7 +709,7 @@ fetch_connect(struct url *url, int af, int verbose)
|
|||||||
if (verbose)
|
if (verbose)
|
||||||
fetch_info("connecting to %s:%d", connurl->host, connurl->port);
|
fetch_info("connecting to %s:%d", connurl->host, connurl->port);
|
||||||
|
|
||||||
sd = happy_eyeballs_connect(res0);
|
sd = happy_eyeballs_connect(res0, verbose);
|
||||||
freeaddrinfo(res0);
|
freeaddrinfo(res0);
|
||||||
if (sd == -1)
|
if (sd == -1)
|
||||||
return (NULL);
|
return (NULL);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user