apps/docs/content/troubleshooting/edge-function-504-error-response.mdx
An internal 504 from an Edge Function means the function failed to initiate a response within 150s, the max execution time.
As of now, this limit cannot be increased. If your function always needs more time than allowed, skip to When optimization isn't enough section. Otherwise, follow the steps below to speed up response times.
You can filter for 504 events in the Log Explorer with the below query:
select
cast(timestamp as datetime) as timestamp,
req.pathname,
res.status_code,
metadata.execution_time_ms
from
function_edge_logs
cross join UNNEST(metadata) as metadata
cross join UNNEST(metadata.request) as req
cross join UNNEST(metadata.response) as res
where res.status_code = 504
limit 20;
You can further explore how much time a specific function takes on average by running the below query:
select
req.pathname,
res.status_code,
AVG(metadata.execution_time_ms) as avg_runtime_ms,
MIN(metadata.execution_time_ms) as min_runtime_ms,
MAX(metadata.execution_time_ms) as max_runtime_ms
from
function_edge_logs
cross join UNNEST(metadata) as metadata
cross join UNNEST(metadata.request) as req
cross join UNNEST(sb) as sb
cross join UNNEST(req.headers) as headers
cross join UNNEST(metadata.response) as res
where req.pathname = '/functions/v1/YOUR_FUNCTION_NAME' -- <---add your function name or remove filter
group by req.pathname, res.status_code
limit 20;
Once candidate functions are identified, you can investigate more thoroughly.
Add console.time labels around suspicious sections to pinpoint where time is being spent:
console.time('fetch-user-data')
const userData = await fetchUserFromDatabase(id)
console.timeEnd('fetch-user-data')
The time response will show up in the function's dashboard under the Log tab with the label you assigned it. You can use the label as a keyword to search for the timestamp.
For local development, you can also step through execution with Chrome Dev Tools.
Common culprits:
The below suggestions are some strategies you can pursue to reduce execution times.
If requests are independent of one another, rather than calling them sequentially, you can speed up operations by calling them in parallel with Promise.all:
// Before: sequential: total time = A + B
const resultA = await fetch('https://api-one.com/data')
const resultB = await fetch('https://api-two.com/data')
// After: parallel: total time = max(A, B)
const [resultA, resultB] = await Promise.all([
fetch('https://api-one.com/data'),
fetch('https://api-two.com/data'),
])
The Edge Function may be doing more than it needs to. It may be better to split up it's logic into multiple parts that can be called individually and run for shorter periods.
You can initiate functions in a background task to be handled independently of the request/response handler. The process is outlined in the Background Task docs
Instead of managing all operations within the function itself, there may be an external API that can execute jobs faster and then send back results. Alternatively, you may be able to offload some processing to the requester rather than doing everything within the function itself.
Some libraries, may be more optimal than others. It's worth exploring alternative libraries that may execute faster or are lighter weight.
If you are connecting to Supabase Postgres from your function, you can inspect the speed of your queries in the performance dashboard. If the operations are slow, you can review performance and index advisor for improvement suggestions.
If you are making requests to the same resource and the return values change irregularly, you may be able to implement an external cache layer, such as Upstash Redis to reduce wait times.
If the function is used to evaluate user submissions, you can restrict load size to reduce computational time. You can use the below query in the log explorer to filter by the content size in bytes provided by the initial requester:
select
req.pathname as function_name,
res.status_code,
AVG(COALESCE(cast(headers.content_length as int), 0)) as avg_content_size_in_bytes,
MIN(COALESCE(cast(headers.content_length as int), 0)) as min_content_size_in_bytes,
MAX(COALESCE(cast(headers.content_length as int), 0)) as max_content_size_in_bytes
from
function_edge_logs
cross join UNNEST(metadata) as metadata
cross join UNNEST(metadata.request) as req
cross join UNNEST(sb) as sb
cross join UNNEST(req.headers) as headers
cross join UNNEST(metadata.response) as res
where req.pathname = '/functions/v1/induce-504'
group by req.pathname, res.status_code
limit 10;
Edge Functions have a hard runtime ceiling that cannot be raised. If your use case genuinely requires more, consider:
Here are some further resources you can review