Create a serverless, globally distributed time-series API with Timescale
In this tutorial, you will learn to build an API on Workers which will ingest and query time-series data stored in Timescale ↗ (they make PostgreSQL faster in the cloud).
You will create and deploy a Worker function that exposes API routes for ingesting data, and use Hyperdrive ↗ to proxy your database connection from the edge and maintain a connection pool to prevent us having to make a new database connection on every request.
You will learn how to:
- Build and deploy a Cloudflare Worker.
- Use Worker secrets with the Wrangler CLI.
- Deploy a Timescale database service.
- Connect your Worker to your Timescale database service with Hyperdrive.
- Query your new API.
You can learn more about Timescale by reading their documentation ↗.
Run the following command to create a Worker project from the command line:
For 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).
Make note of the URL that your application was deployed to. You will be using it when you configure your GitHub webhook.
Change into the directory you just created for your Worker project:
If you are creating a new service, go to the Timescale Console ↗ and follow these steps:
- Select Create Service by selecting the black plus in the upper right.
- Choose Time Series as the service type.
- Choose your desired region and instance size. 1 CPU will be enough for this tutorial.
- Set a service name to replace the randomly generated one.
- Select Create Service.
- On the right hand side, expand the Connection Info dialog and copy the Service URL.
- Copy the password which is displayed. You will not be able to retrieve this again.
- Select I stored my password, go to service overview.
If you are using a service you created previously, you can retrieve your service connection information in the Timescale Console ↗:
- Select the service (database) you want Hyperdrive to connect to.
- Expand Connection info.
- Copy the Service URL. The Service URL is the connection string that Hyperdrive will use to connect. This string includes the database hostname, port number and database name.
Insert your password into the Service URL as follows (leaving the portion after the @ untouched):
This will be referred to as SERVICEURL in the following sections.
Timescale allows you to convert regular PostgreSQL tables into hypertables ↗, tables used to deal with time-series, events, or analytics data. Once you have made this change, Timescale will seamlessly manage the hypertable's partitioning, as well as allow you to apply other features like compression or continuous aggregates.
Connect to your Timescale database using the Service URL you copied in the last step (it has the password embedded).
If you are using the default PostgreSQL CLI tool psql ↗ to connect, you would run psql like below (substituting your Service URL from the previous step). You could also connect using a graphical tool like PgAdmin ↗.
Once you are connected, create your table by pasting the following SQL:
Timescale will manage the rest for you as you ingest and query data.
To create a new Hyperdrive instance you will need:
- Your SERVICEURL from step 2.
- A name for your Hyperdrive service. For this tutorial, you will use hyperdrive.
Hyperdrive uses the create
command with the --connection-string
argument to pass this information. Run it as follows:
This command outputs your Hyperdrive ID. You can now bind your Hyperdrive configuration to your Worker in your wrangler.toml
configuration by replacing the content with the following:
Install the Postgres driver into your Worker project:
Now copy the below Worker code, and replace the current code in ./src/index.ts
. The code below:
- Uses Hyperdrive to connect to Timescale using the connection string generated from
env.HYPERDRIVE.connectionString
directly to the driver. - Creates a
POST
route which accepts an array of JSON readings to insert into Timescale in one transaction. - Creates a
GET
route which takes alimit
parameter and returns the most recent readings. This could be adapted to filter by ID or by timestamp.
Run the following command to redeploy your Worker:
Your application is now live and accessible at timescale-api.<YOUR_SUBDOMAIN>.workers.dev
. The exact URI will be shown in the output of the wrangler command you just ran.
After deploying, you can interact with your Timescale IoT readings database using your Cloudflare Worker. Connection from the edge will be faster because you are using Cloudflare Hyperdrive to connect from the edge.
You can now use your Cloudflare Worker to insert new rows into the readings
table. To test this functionality, send a POST
request to your Worker’s URL with the /readings
path, along with a JSON payload containing the new product data:
This tutorial omits the ts
(the timestamp) and metadata
(the JSON blob) so they will be set to now()
and NULL
respectively.
Once you have sent the POST
request you can also issue a GET
request to your Worker’s URL with the /readings
path. Set the limit
parameter to control the amount of returned records.
If you have curl installed you can test with the following commands (replace <YOUR_SUBDOMAIN>
with your subdomain from the deploy command above):
In this tutorial, you have learned how to create a working example to ingest and query readings from the edge with Timescale, Workers, Hyperdrive, and TypeScript.
- Learn more about How Hyperdrive Works.
- Learn more about Timescale ↗.
- Refer to the troubleshooting guide to debug common issues.