We've added optional pagination for list endpoints to optimize retrieving data.
When retrieving data, it's important for it to be quick and easy. Our list endpoints have always returned all existing data when called. While this default is convenient, it can lead to performance issues when dealing with large datasets.
Today, we've introduced optional pagination to our existing list endpoints to help you efficiently browse through large datasets when needed.
When calling an existing list endpoint, you can optionally add the limit query parameter to the request.
const resend = new Resend('re_xxxxxxxxx');const contacts = await resend.contacts.list({limit: 30,});
This will return the first 30 contacts in the list.
{"object": "list","has_more": true,"data": [/* Array of 30 contacts */]}
To retrieve additional pages, use the before or after params to go through the list.
To paginate forward through results (newer to older items), use the after parameter with the ID of the last item from the current page:
const resend = new Resend('re_xxxxxxxxx');// First pageconst { data: firstPage } = await resend.contacts.list({ limit: 10 });// Second page (if has_more is true)if (firstPage.has_more) {const lastId = firstPage.data[firstPage.data.length - 1].id;const { data: secondPage } = await resend.contacts.list({limit: 10,after: lastId});}
To paginate backward through results (older to newer items), use the before parameter with the ID of the first item from the current page (or the most recent ID you have in your system):
const resend = new Resend('re_xxxxxxxxx');// Start from a specific point and go backwardconst page = await resend.contacts.list({limit: 10,before: 'some-contact-id'});if (page.data.has_more) {const firstId = page.data.data[0].id;const previousPage = await resend.contacts.list({limit: 10,before: firstId});}
You can safely navigate lists with guaranteed stability, even if new objects are created or deleted while you’re still requesting pages.
The following endpoints support optional pagination:
All paginated endpoints support the following parameters:
limit: The number of items to return per page. Maximum is 100. Minimum is 1.after: The cursor after which to start retrieving items. To get the next page, use the ID of the last item from the current page. This will return the page that starts after the object with this ID (excluding the passed ID itself).before: The cursor before which to start retrieving items. To get the previous page, use the ID of the first item from the current page. This will return the page that ends before the object with this ID (excluding the passed ID itself).Note that for these endpoints the limit parameter is optional. If you do not provide a limit, all available resources will be returned in a single response.
When you request a paginated endpoint, the response includes:
{"object": "list", // Always set to list."has_more": true, // Indicates whether there are more elements available."data": [/* Array of resources */]}
Pagination may return the following validation errors:
validation_error: Invalid cursor format or limit out of range (1-100).validation_error: Both before and after parameters provided.{"name": "validation_error","statusCode": 422,"message": "The pagination limit must be a number between 1 and 100. See https://resend.com/docs/api-reference/pagination for more information."}
Follow these best practices when using pagination:
limit that balances performance and usability. Smaller pages are good for real-time applications, while larger pages (hundreds of items) work better for bulk processing.has_more field before attempting to fetch additional pages. This prevents unnecessary API calls when you’ve reached the end of the dataset.We're excited for the new control pagination gives you over your data and plan to include pagination by default in the future. For more information, check out our pagination documentation.
If you have any questions, please reach out to us and we'll be happy to help.