Open guotie opened 7 months ago
Thanks for reporting!
Can you provide steps to reproduce? We often need a reproducible example, e.g. some code that allows someone else to recreate your problem by just copying and pasting it. If it involves more than a couple of different file, create a new repository on GitHub and add a link to that.
repo is here: https://github.com/guotie/fetch-failed
I think the problem is timeout.
when the network is Ok, it rarely throw fetch failed
; when the network is busy, error occurs more frequent.
Facing a similar issue
same issue, anyone know the solution?
if the internet connectivity was the issue, then why curl always work and never throw timeout?
it seems this error only appear in certain environment or device? it's hard to reproduce but I think the bug really exists
someone already describe the same issue here too https://github.com/nodejs/undici/issues/2990
or in certain host? I faced this issue when trying to fetch telegram api
here's the endpoint that I try to fetch
curl "https://api.telegram.org/bot7003873933:AAFKl0LwWViMJIA34-qjbTh7nZwcNQr2hFs/getFile?file_id=CAACAgEAAxUAAWYT6IXJGTzY4S96PCbyqyO7fBXXAAIJEgACkweVC56njKMcTovTNAQ"
Can you provide an Minimum Reproducible Example to support you better?
I've provided the Minimum Reproducible Example on my Gist, as you suggested.
If you have a strong or reliable internet connection, consider simulating slow connectivity to see if the error replicates. After all, ETIMEDOUT errors are more likely to occur under limited bandwidth conditions.
Interestingly, while fetch sometimes throws this error, curl seems to be able to avoid it in this scenario.
Hmm, this does not seem like an undici
error per se but rather a different way of handling connect timeouts.
The errors shown by the example and the roots of the issue mostly to the initial TCP connection (including TLS), meaning that undici
timed out before the server could finish the initial connect operation.
You can attempt to extend the overall timeout while creating a custom Agent
(See https://undici.nodejs.org/#/docs/api/Client?id=parameter-clientoptions) and test if that solves the timeout issue, which seems related directly to the network conditions.
As well, you can wrap it with the RetryAgent
to automatically retry upon this errors.
in the docs it says:
bodyTimeout - Defaults to 300 seconds.
headersTimeout - Defaults to 300 seconds.
why it close the connection too early if the default was 300 seconds? does the node fetch use undici differently in the internal?
btw, this is unrelated question: why I can't access undici directly (require('undici')
) if it was used in fetch implementation of node? is there a way to expose it, so I can use the RetryAgent
class without installing undici dependency?
The timeouts you mention are applied directly at http
level, meanwhile the timeout
I'm referring to, is linked to the TCP handshake; which by default is 10s
(lower than the body and headers).
Sadly no, you'll need to install undici
to make use of the RetryAgent
.
Thanks for your assistance! It's confirmed that increasing the connection timeout resolved the issue in my scenario ^^
I am able to reliably repro it when fetching too many urls at once. https://github.com/nodejs/node-core-utils/issues/810
I've been working on the same problem for 1 day. Some comments talk about connection, but I don't think that's the case, because I have a very good connection, and it's not a device problem.
check your ip dns configuration or clear your dns cache; make sure your router is serving a real dns server, or change your ip dns to target a real dns server.
The observation is that on my online server the code doesn't throw this AggregateError [ETIMEDOUT] error, but on my local machine it always throws this error only when I'm at home on my local network.
Looking further, the main errors thrown are ETIMEDOUT and ENETUNREACH. This usually occurs when the DNS module is unable to resolve the IP address.
node:internal/deps/undici/undici:11754
Error.captureStackTrace(err, this);
^
TypeError: fetch failed
at node:internal/deps/undici/undici:11754:11
at process.processTicksAndRejections (node:internal/process/task_queues:95:5) {
cause: AggregateError [ETIMEDOUT]:
at internalConnectMultiple (node:net:1114:18)
at internalConnectMultiple (node:net:1177:5)
at Timeout.internalConnectMultipleTimeout (node:net:1687:3)
at listOnTimeout (node:internal/timers:575:11)
at process.processTimers (node:internal/timers:514:7) {
code: 'ETIMEDOUT',
[errors]: [
Error: connect ETIMEDOUT <IP:PORT>
at createConnectionError (node:net:1634:14)
at Timeout.internalConnectMultipleTimeout (node:net:1685:38)
at listOnTimeout (node:internal/timers:575:11)
at process.processTimers (node:internal/timers:514:7) {
errno: -110,
code: 'ETIMEDOUT',
syscall: 'connect',
address: '3.125.177.232',
port: 443
},
Error: connect ENETUNREACH <IP6:PORT> - Local (:::0)
at internalConnectMultiple (node:net:1176:40)
at Timeout.internalConnectMultipleTimeout (node:net:1687:3)
at listOnTimeout (node:internal/timers:575:11)
at process.processTimers (node:internal/timers:514:7) {
errno: -101,
code: 'ENETUNREACH',
syscall: 'connect',
address: '64:ff9b::37d:b1e8',
port: 443
}
]
}
}
if the request got succesful once, it seems that the ip address in the log error can be the cached ones stored in your machine before.
My hypothesis is that when working under a network or router with a non-dns server, the node's dns module is no longer able to evaluate the ip address and doesn't use the cached one and sends errors.
I'll check this out, but you have the solution. I changed my dns configuration in /etc/resolv.conf to 8.8.8.8 or 1.1.1.1. and everything's back to normal!
Bug Description
my code is typescript. I run it in two ways:
when use bun run it, no error occurs; when use node run it, it occurs fetch failed extremely frequent
the request is an apollo graphql request. proxy is local http proxy:
Logs & Screenshots
Environment
Mac M1 Sonoma 14.2
Nodejs v21
Additional context