Paginate cross-product query results with the Azure Cosmos DB for NoSQL SDK
Azure Cosmos DB queries will typically have multiple pages of results. Pagination is done automatically server-side when Azure Cosmos DB cannot return all query results in one single execution. In many applications, you will want to write code using the SDK to process your query results in batches in a performant manner.
In this lab, you’ll create a feed iterator that can be used in a loop to iterate over your entire result set.
Prepare your development environment
If you have not already cloned the lab code repository for Build copilots with Azure Cosmos DB and set up your local environment, view the Setup local lab environment instructions to do so.
Create an Azure Cosmos DB for NoSQL account
If you already created an Azure Cosmos DB for NoSQL account for the Build copilots with Azure Cosmos DB labs on this site, you can use it for this lab and skip ahead to the next section. Otherwise, view the Setup Azure Cosmos DB instructions to create an Azure Cosmos DB for NoSQL account that you will use throughout the lab modules and grant your user identity access to manage data in the account by assigning it to the Cosmos DB Built-in Data Contributor role.
Create Azure Cosmos DB database and container with sample data
If you already created an Azure Cosmos DB database named cosmicworks-full and container within it named products, which is preloaded with sample data, you can use it for this lab and skip ahead to the next section. Otherwise, follow the steps below to create a new sample database and container.
Click to expand/collapse steps to create database and container with sample data
-
Within the newly created Azure Cosmos DB account resource, navigate to the Data Explorer pane.
-
In the Data Explorer, select Launch quick start on the home page.
-
Within the New Container form, enter the following values:
-
Database id:
cosmicworks-full
-
Container id:
products
-
Partition key:
/categoryId
-
Analytical store:
Off
-
Database id:
-
Select OK to create the new container. This process will take a minute or two while it creates the resources and preloads the container with sample product data.
-
Keep the browser tab open, as we will return to it later.
-
Switch back to Visual Studio Code.
Import the @azure/cosmos library
The @azure/cosmos library is available on npm for easy installation into your JavaScript projects.
-
In Visual Studio Code, in the Explorer pane, browse to the javascript/06-sdk-pagination folder.
-
Open the context menu for the javascript/06-sdk-pagination folder and then select Open in Integrated Terminal to open a new terminal instance.
📝 This command will open the terminal with the starting directory already set to the javascript/06-sdk-pagination folder.
-
Initialize a new Node.js project:
npm init -y
-
Install the @azure/cosmos package using the following command:
npm install @azure/cosmos
-
Install the @azure/identity library, which allows us to use Azure authentication to connect to the Azure Cosmos DB workspace, using the following command:
npm install @azure/identity
Iterate over the results of a SQL query using the SDK
When processing query results, you must make sure your code progresses through all pages of results and checks to see if any more pages are remaining before making subsequent requests.
-
In Visual Studio Code, in the Explorer pane, browse to the javascript/06-sdk-pagination folder.
-
Open the empty JavaScript file named script.js.
-
Add the following
require
statements to import the @azure/cosmos and @azure/identity libraries:const { CosmosClient } = require("@azure/cosmos"); const { DefaultAzureCredential } = require("@azure/identity"); process.env.NODE_TLS_REJECT_UNAUTHORIZED = 0
-
Add variables named endpoint and credential and set the endpoint value to the endpoint of the Azure Cosmos DB account you created earlier. The credential variable should be set to a new instance of the DefaultAzureCredential class:
const endpoint = "<cosmos-endpoint>"; const credential = new DefaultAzureCredential();
📝 For example, if your endpoint is: https://dp420.documents.azure.com:443/, the statement would be: const endpoint = “https://dp420.documents.azure.com:443/”;.
-
Add a new variable named client and initialize it as a new instance of the CosmosClient class using the endpoint and credential variables:
const client = new CosmosClient({ endpoint, aadCredentials: credential });
-
Create a new method named paginateResults and code to execute that method when you run the script. You will add the code to query the container within this method:
async function paginateResults() { // Query the container } paginateResults().catch((error) => { console.error(error); });
-
Inside the paginateResults method, add the following code to connect to the database and container you created earlier::
const database = client.database("cosmicworks-full"); const container = database.container("products");
-
Create a query string variable named
sql
with a value ofSELECT * FROM products p
.const sql = "SELECT * FROM products p";
-
Create a new variable named
options
and set it to an object with theenableCrossPartitionQuery
property set totrue
. This property allows sending more than one request to execute the query in the Azure Cosmos DB service. More than one request is necessary if the query is not scoped to a single partition key value. Set themaxItemCount
property to50
to limit the number of items per page:const options = { enableCrossPartitionQuery: true, maxItemCount: 50 // Set the maximum number of items per page };
-
Invoke the
query
method with thesql
andoptions
variables as parameters to the constructor. This method returns an iterator that can be used to fetch the next page of results:const iterator = container.items.query(query, options);
-
Iterate over the paginated results and print the
id
,name
, andprice
of each item. Theiterator.getAsyncIterator
method returns an async iterator that can be used to fetch thefor await...of
loop to retrieve each page of results. This loop automatically handles the asynchronous iteration over the pages.for await (const page of iterator.getAsyncIterator()) { page.resources.forEach(product => { console.log(`[${product.id}] ${product.name} $${product.price.toFixed(2)}`); }); }
-
Your script.js file should now look like this:
const { CosmosClient } = require("@azure/cosmos"); const { DefaultAzureCredential } = require("@azure/identity"); process.env.NODE_TLS_REJECT_UNAUTHORIZED = 0 const endpoint = "<cosmos-endpoint>"; const credential = new DefaultAzureCredential(); const client = new CosmosClient({ endpoint, aadCredentials: credential }); async function paginateResults() { const database = client.database("cosmicworks-full"); const container = database.container("products"); const query = "SELECT * FROM products WHERE products.price > 500"; const options = { enableCrossPartitionQuery: true, maxItemCount: 50 // Set the maximum number of items per page }; const iterator = container.items.query(query, options); for await (const page of iterator.getAsyncIterator()) { page.resources.forEach(product => { console.log(`[${product.id}] ${product.name} $${product.price.toFixed(2)}`); }); } } paginateResults().catch((error) => { console.error(error); });
-
Save the script.js file.
-
Before running the script, you must log into Azure using the
az login
command. At the terminal window, run:az login
-
Run the script to create the database and container:
node script.js
-
The script will now output pages of 50 items at a time.
💡 The query will match hundreds of items in the products container.
-
Close the integrated terminal.
-
Close Visual Studio Code.