Get started
This guide instructs you through:
- Creating your first database using D1, Cloudflare's native serverless SQL database.
- Creating a schema and querying your database via the command-line.
- Connecting a Cloudflare Worker to your D1 database to query your D1 database programmatically.
You can perform these tasks through the CLI or through the Cloudflare dashboard.
- Sign up for a Cloudflare account β.
- Install Node.jsβ.
Node.js version manager
 Use a Node version manager like Volta β or nvm β to avoid permission issues and change Node.js versions. Wrangler, discussed later in this guide, requires a Node version of 16.17.0 or later.
Create a new Worker as the means to query your database.
- 
Create a new project named d1-tutorialby running:Terminal window npm create cloudflare@latest -- d1-tutorialTerminal window pnpm create cloudflare@latest d1-tutorialTerminal window yarn create cloudflare d1-tutorialFor setup, select the following options: - For What would you like to start with?, choose Hello World example.
- For Which template would you like to use?, choose Hello World Worker.
- For Which language do you want to use?, choose TypeScript.
- For Do you want to use git for version control?, choose Yes.
- For Do you want to deploy your application?, choose No(we will be making some changes before deploying).
 This creates a new d1-tutorialdirectory as illustrated below.- Directoryd1-tutorial- Directorynode_modules/- β¦
 
- Directorytest/- β¦
 
- Directorysrc- index.ts
 
- package-lock.json
- package.json
- testconfig.json
- vitest.config.mts
- worker-configuration.d.ts
- wrangler.jsonc
 
 Your new d1-tutorialdirectory includes:- A "Hello World"Worker inindex.ts.
- A [Wrangler configuration file](/workers/wrangler/configuration/). This file is how your d1-tutorialWorker accesses your D1 database.
 
- For What would you like to start with?, choose 
- Log in to your Cloudflare dashboard β and select your account.
- Go to your account > Workers & Pages > Overview.
- Select Create.
- Select Create Worker.
- Name your Worker. For this tutorial, name your Worker d1-tutorial.
- Select Deploy.
A D1 database is conceptually similar to many other databases: a database may contain one or more tables, the ability to query those tables, and optional indexes. D1 uses the familiar SQL query language β (as used by SQLite).
To create your first D1 database:
- 
Change into the directory you just created for your Workers project: Terminal window cd d1-tutorial
- 
Run the following wrangler d1command and give your database a name. In this tutorial, the database is namedprod-d1-tutorial:Terminal window npx wrangler d1 create prod-d1-tutorialβ Successfully created DB 'prod-d1-tutorial'[[d1_databases]]binding = "DB" # available in your Worker on env.DBdatabase_name = "prod-d1-tutorial"database_id = "<unique-ID-for-your-database>"
This creates a new D1 database and outputs the binding configuration needed in the next step.
- Go to Storage & Databases > D1.
- Select Create.
- Name your database. For this tutorial, name your D1 database prod-d1-tutorial.
- (Optional) Provide a location hint. Location hint is an optional parameter you can provide to indicate your desired geographical location for your database. Refer to Provide a location hint for more information.
- Select Create.
You must create a binding for your Worker to connect to your D1 database. Bindings allow your Workers to access resources, like D1, on the Cloudflare developer platform.
To bind your D1 database to your Worker:
You create bindings by updating your Wrangler file.
- 
Copy the lines obtained from step 2 from your terminal. 
- 
Add them to the end of your Wrangler file. {"d1_databases": [{"binding": "DB","database_name": "prod-d1-tutorial","database_id": "<unique-ID-for-your-database>"}]}[[d1_databases]]binding = "DB" # available in your Worker on env.DBdatabase_name = "prod-d1-tutorial"database_id = "<unique-ID-for-your-database>"Specifically: - The value (string) you set for bindingis the binding name, and is used to reference this database in your Worker. In this tutorial, name your bindingDB.
- The binding name must be a valid JavaScript variable name β. For example, binding = "MY_DB"orbinding = "productionDB"would both be valid names for the binding.
- Your binding is available in your Worker at env.<BINDING_NAME>and the D1 Workers Binding API is exposed on this binding.
 
- The value (string) you set for 
You can also bind your D1 database to a Pages Function. For more information, refer to Functions Bindings for D1.
You create bindings by adding them to the Worker you have created.
- 
Go to Workers & Pages > Overview. 
- 
Select the d1-tutorialWorker you created in step 1.
- 
Select Settings. 
- 
Scroll to Bindings, then select Add. 
- 
Select D1 database. 
- 
Name your binding in Variable name, then select the prod-d1-tutorialD1 database you created in step 2 from the dropdown menu. For this tutorial, name your bindingDB.
- 
Select Deploy to deploy your binding. When deploying, there are two options: - Deploy: Immediately deploy the binding to 100% of your audience.
- Save version: Save a version of the binding which you can deploy in the future.
 For this tutorial, select Deploy. 
After correctly preparing your Wrangler configuration file, set up your database. Use the example schema.sql file below to initialize your database.
- 
Copy the following code and save it as a schema.sqlfile in thed1-tutorialWorker directory you created in step 1:DROP TABLE IF EXISTS Customers;CREATE TABLE IF NOT EXISTS Customers (CustomerId INTEGER PRIMARY KEY, CompanyName TEXT, ContactName TEXT);INSERT INTO Customers (CustomerID, CompanyName, ContactName) VALUES (1, 'Alfreds Futterkiste', 'Maria Anders'), (4, 'Around the Horn', 'Thomas Hardy'), (11, 'Bs Beverages', 'Victoria Ashworth'), (13, 'Bs Beverages', 'Random Name');
- 
Initialize your database to run and test locally first. Bootstrap your new D1 database by running: Terminal window npx wrangler d1 execute prod-d1-tutorial --local --file=./schema.sql
- 
Validate your data is in your database by running: Terminal window npx wrangler d1 execute prod-d1-tutorial --local --command="SELECT * FROM Customers"π Mapping SQL input into an array of statementsπ Executing on local database production-db-backend (5f092302-3fbd-4247-a873-bf1afc5150b) from .wrangler/state/v3/d1:ββββββββββββββ¬ββββββββββββββββββββββ¬βββββββββββββββββββββ CustomerId β CompanyName β ContactName βββββββββββββββΌββββββββββββββββββββββΌββββββββββββββββββββ€β 1 β Alfreds Futterkiste β Maria Anders βββββββββββββββΌββββββββββββββββββββββΌββββββββββββββββββββ€β 4 β Around the Horn β Thomas Hardy βββββββββββββββΌββββββββββββββββββββββΌββββββββββββββββββββ€β 11 β Bs Beverages β Victoria Ashworth βββββββββββββββΌββββββββββββββββββββββΌββββββββββββββββββββ€β 13 β Bs Beverages β Random Name βββββββββββββββ΄ββββββββββββββββββββββ΄ββββββββββββββββββββ
Use the Dashboard to create a table and populate it with data.
- 
Go to Storage & Databases > D1. 
- 
Select the prod-d1-tutorialdatabase you created in step 2.
- 
Select Console. 
- 
Paste the following SQL snippet. DROP TABLE IF EXISTS Customers;CREATE TABLE IF NOT EXISTS Customers (CustomerId INTEGER PRIMARY KEY, CompanyName TEXT, ContactName TEXT);INSERT INTO Customers (CustomerID, CompanyName, ContactName) VALUES (1, 'Alfreds Futterkiste', 'Maria Anders'), (4, 'Around the Horn', 'Thomas Hardy'), (11, 'Bs Beverages', 'Victoria Ashworth'), (13, 'Bs Beverages', 'Random Name');
- 
Select Execute. This creates a table called Customersin yourprod-d1-tutorialdatabase.
- 
Select Tables, then select the Customerstable to view the contents of the table.
After you have set up your database, run an SQL query from within your Worker.
- 
Navigate to your d1-tutorialWorker and open theindex.tsfile. Theindex.tsfile is where you configure your Worker's interactions with D1.
- 
Clear the content of index.ts.
- 
Paste the following code snippet into your index.tsfile:index.js export default {async fetch(request, env) {const { pathname } = new URL(request.url);if (pathname === "/api/beverages") {// If you did not use `DB` as your binding name, change it hereconst { results } = await env.DB.prepare("SELECT * FROM Customers WHERE CompanyName = ?",).bind("Bs Beverages").all();return Response.json(results);}return new Response("Call /api/beverages to see everyone who works at Bs Beverages",);},};index.ts export interface Env {// If you set another name in the Wrangler config file for the value for 'binding',// replace "DB" with the variable name you defined.DB: D1Database;}export default {async fetch(request, env): Promise<Response> {const { pathname } = new URL(request.url);if (pathname === "/api/beverages") {// If you did not use `DB` as your binding name, change it hereconst { results } = await env.DB.prepare("SELECT * FROM Customers WHERE CompanyName = ?",).bind("Bs Beverages").all();return Response.json(results);}return new Response("Call /api/beverages to see everyone who works at Bs Beverages",);},} satisfies ExportedHandler<Env>;In the code above, you: - Define a binding to your D1 database in your TypeScript code. This binding matches the bindingvalue you set in the Wrangler configuration file under[[d1_databases]].
- Query your database using env.DB.prepareto issue a prepared query with a placeholder (the?in the query).
- Call bind()to safely and securely bind a value to that placeholder. In a real application, you would allow a user to define theCompanyNamethey want to list results for. Usingbind()prevents users from executing arbitrary SQL (known as "SQL injection") against your application and deleting or otherwise modifying your database.
- Execute the query by calling all()to return all rows (or none, if the query returns none).
- Return your query results, if any, in JSON format with Response.json(results).
 
- Define a binding to your D1 database in your TypeScript code. This binding matches the 
After configuring your Worker, you can test your project locally before you deploy globally.
You can query your D1 database using your Worker.
- 
Go to Workers & Pages > Overview. 
- 
Select the d1-tutorialWorker you created.
- 
Select Edit Code. 
- 
Clear the contents of the worker.jsfile, then paste the following code:export default {async fetch(request, env) {const { pathname } = new URL(request.url);if (pathname === "/api/beverages") {// If you did not use `DB` as your binding name, change it hereconst { results } = await env.DB.prepare("SELECT * FROM Customers WHERE CompanyName = ?").bind("Bs Beverages").all();return new Response(JSON.stringify(results), {headers: { 'Content-Type': 'application/json' }});}return new Response("Call /api/beverages to see everyone who works at Bs Beverages");},};
- 
Select Save. 
Deploy your database on Cloudflare's global network.
To deploy your Worker to production using Wrangler, you must first repeat the database configuration steps after replacing the --local flag with the --remote flag to give your Worker data to read. This creates the database tables and imports the data into the production version of your database.
- 
Bootstrap your database with the schema.sqlfile you created in step 4:Terminal window npx wrangler d1 execute prod-d1-tutorial --remote --file=./schema.sql
- 
Validate the data is in production by running: Terminal window npx wrangler d1 execute prod-d1-tutorial --remote --command="SELECT * FROM Customers"
- 
Deploy your Worker to make your project accessible on the Internet. Run: Terminal window npx wrangler deployOutputs: https://d1-tutorial.<YOUR_SUBDOMAIN>.workers.devYou can now visit the URL for your newly created project to query your live database. For example, if the URL of your new Worker is d1-tutorial.<YOUR_SUBDOMAIN>.workers.dev, accessinghttps://d1-tutorial.<YOUR_SUBDOMAIN>.workers.dev/api/beveragessends a request to your Worker that queries your live database directly.
- 
Test your database is running successfully. Add /api/beveragesto the provided Wrangler URL. For example,https://d1-tutorial.<YOUR_SUBDOMAIN>.workers.dev/api/beverages.
- Go to Workers & Pages > Overview.
- Select your d1-tutorialWorker.
- Select Deployments.
- From the Version History table, select Deploy version.
- From the Deploy version page, select Deploy.
This deploys the latest version of the Worker code to production.
If you are using D1 with Wrangler, you can test your database locally. While in your project directory:
- 
Run wrangler dev:Terminal window npx wrangler devWhen you run wrangler dev, Wrangler provides a URL (most likelylocalhost:8787) to review your Worker.
- 
Go to the URL. The page displays Call /api/beverages to see everyone who works at Bs Beverages.
- 
Test your database is running successfully. Add /api/beveragesto the provided Wrangler URL. For example,localhost:8787/api/beverages.
If successful, the browser displays your data.
To delete your database:
Run:
npx wrangler d1 delete prod-d1-tutorial- 
Go to Storages & Databases > D1. 
- 
Select your prod-d1-tutorialD1 database.
- 
Select Settings. 
- 
Select Delete. 
- 
Type the name of the database ( prod-d1-tutorial) to confirm the deletion.
If you want to delete your Worker:
Run:
npx wrangler delete d1-tutorial- 
Go to Workers & Pages > Overview. 
- 
Select your d1-tutorialWorker.
- 
Select Settings. 
- 
Scroll to the bottom of the page, then select Delete. 
- 
Type the name of the Worker ( d1-tutorial) to confirm the deletion.
In this tutorial, you have:
- Created a D1 database
- Created a Worker to access that database
- Deployed your project globally
If you have any feature requests or notice any bugs, share your feedback directly with the Cloudflare team by joining the Cloudflare Developers community on Discord β.
- See supported Wrangler commands for D1.
- Learn how to use D1 Worker Binding APIs within your Worker, and test them from the API playground.
- Explore community projects built on D1.