Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/velsa/notion-ts-client

Generates an easy to use and fully typed API to access and modify the data in Notion Databases
https://github.com/velsa/notion-ts-client

Last synced: 2 months ago
JSON representation

Generates an easy to use and fully typed API to access and modify the data in Notion Databases

Awesome Lists containing this project

README

        

# Notion Typescript Client

Generates an easy to use and fully typed API client to access and modify the data in your Notion Databases!

## Supports:

- Database query with fully typed filters and sorts
- Fully typed database page methods (CRUD)
- Built-in rate limiting for Notion API calls
- Quick start with an easy to use CLI: `npx notion-ts-client`
- Customize variable and type names using JSON config file
- Define read-only properties via config file for additional safety

## What can you do with it?

Create complex calculations and intelligent automations for your Notion Databases using Typescript, ensuring complete type safety. Use automagically generated SDK with Custom Types per Database to safely read from and write to your Notion databases

- Use **notion-ts-client** with Webhooks for building powerful automations and formulas (see example below)
- Use Notion as your Database / Backoffice and **notion-ts-client** will generate custom Typescript SDKs for your code. Ensure complete type equality between your backend and frontend.


## Demo:

```ts
// Let's imagine you have a database of online events in your Notion :)
import { OnlineEventsDatabase, OnlineEventsResponseDTO, OnlineEventsPatchDTO } from 'notion-sdk/dbs/online-events'

// Use custom generated database class to work with your DB
const db = new OnlineEventsDatabase({
notionSecret: process.env.MY_NOTION_SECRET,
})

// Query the Notion DB using fully typed filter and sorts
const queryResponse = await db.query({
filter: {
and: [
{ type: { contains: 'Webinar' } }, // <--- type safe!
{ organization: { equals: 'My Org' } }, // <--- type safe!
],
},
sorts: [{ property: 'name', direction: 'ascending' }], // <--- type safe!
})

// Access your page properties via custom generated ResponseDTO (Data Transfer Object)
const pages = queryResponse.results.map((r) => new OnlineEventsResponseDTO(r))

console.log(pages[0].properties.organization) // <--- type safe!

// Update your Notion DB via a fully typed custom PatchDTO
// Note: for your convenience readOnly properties are not available on PatchDTO
const pageUpdate = new OnlineEventsPatchDTO({
properties: {
organization: 'Some org',
},
})

await db.updatePage(pages[0].id, pageUpdate)
```

The code looks nice, right? But there is much more under the hood.
ALL of the above code is FULLY TYPED, meaning that every property has a custom type which is directly linked to the configuration of your Notion database.


## How to use:

### Create a new integration in Notion

Visit: https://www.notion.so/my-integrations

Click `Create a new integration` and give it any name you like, e.g. `NotionSDK`.

Copy the secret token.

### Add databases to your integration

Add the databases you want to work with to your integration in Notion:

1. Go to the database page in Notion
2. Click `...` at the top right corner
3. `...` => `Add connections` => The name of your integration

### Generate config file based on your integrations

Run the `init` command:

```sh
npx notion-ts-client@latest init --secret
```

A config file will be generated for all the databases you added to your integration. Generated configuration reflects your database and its properties. You can read about the structure of Notion databases and the available property types in the Notion docs

Here is an example of a generated config file:

```ts
// notion-ts-client.config.json
{
"databases": {
"70b2b25b-7f13-4306-b56a-9486f01efced": {
"_name": "🎫 Online Events", // The name of your Database in Notion (don't change)
"varName": "onlineEvents", // The base name used to generate custom types for your Database
"pathName": "online-events", // Path of your database in SDK folder (/dbs/online-events)
"properties": { // You can change "varName" and "readOnly" fields for each property
// This change will be reflected in the generated SDK
"C%3DP_": { // Notion property ID (don't change)
"_name": "Type", // Notion property name (don't change)
"_type": "multi_select", // Notion property type (don't change)
"varName": "type", // variable name to be generated in the SDK
"readOnly": false // read only properties will not be available in DTOs used for writes
},
"%3D%3DNA": {
"_name": "Organization",
"_type": "select",
"varName": "organization",
"readOnly": false
},
"%3DK%3F%3A": {
"_name": "Short Description",
"_type": "rich_text",
"varName": "shortDescription",
"readOnly": false
},
// ...
```

Once you are done reviewing the config file and editing the `varName` and `readOnly` fields for the properties, you can continue to SDK generation.

### Generate Typescript Clients (SDK)

When you run the `generate` command, **notion-ts-client** automatically updates your config file based on the current database configuration in Notion and generates Typescript Clients (SDKs). Those clients export custom Typescript types and API methods for each database.

Run the `generate` command:

```sh
npx notion-ts-client@latest generate --secret --sdk
```

The Typescript Clients will be generated in the specified directory.

You can now access your Notion Databases in a fullproof and typesafe manner. As you should!

> [!NOTE]
> Every time **notion-ts-client** detects that a new database was added to your integration,
> it will ask you if you want to generate the config and the SDK client for it.
> If you answer No - the database will be added to the ignore list.

### Environment variables

You can also configure your Notion secret and other cli options via environment variables. **notion-ts-client** uses `dotenv/config` to read the environment, so you can create `.env` file in your project root and put all cli config there:

```sh
NOTION_TS_CLIENT_NOTION_SECRET=
NOTION_TS_CLIENT_CONFIG_PATH=./notion-ts-client.config.json
NOTION_TS_CLIENT_SDK_PATH=./src/notion-sdk
```

Now you can simply run

```sh
npx notion-ts-client@latest generate
```

to regenerate your SDKs whenever you need to reflect in your code the changes that were made in Notion.


## Using notion-ts-client with webhooks

To get a webhook every time the data in your Notion database changes you can use this service:

https://notion.hostedhooks.com/

It's a paid service and it costs $10 a month. But it is way better than services like Zapier and will give you maximum level of control over your data.

You can utilize **notion-ts-client** by simply calling `new YourCustomResponseDTO(payload)` in your webhook handler. This way you will get all the properties of a changed page with types and an easy to use API.

Create ANY kind of automations and advanced formulas using the power of Typescript! 😎

### Webhook example

Using Cloudflare Workers:

```ts
import {
OnlineEventsDatabase,
OnlineEventsResponse,
OnlineEventsResponseDTO,
OnlineEventsPatchDTO,
} from 'notion-sdk/dbs/online-events'

export default {
async fetch(request: Request, env: Env): Promise {
if (request.method === 'POST') {
const payload: OnlineEventsResponse = await request.json()
const props = new OnlineEventsResponseDTO(payload).properties

// process page properties...

// update notion page with new properties
const db = new OnlineEventsDatabase({ notionSecret: env.MY_NOTION_SECRET })

await db.updatePage(
payload.id,
new OnlineEventsPatchDTO({
properties: {
// calculated properties
},
}),
)
}
},
}
```


## Important notes for usage in production:

### If a property in your Notion database has been **renamed**:

The methods `query` and `updatePage` will not be affected, since under the hood those methods call Notion API using property IDs. Unfortunately, the `getPage` method can fail, since Notion API returns names of Notion properties and **notion-ts-client** then tries to map property names to var names.

### If a property in Notion has been **removed** or its **type has been changed**:

This might present a problem for the production code. In order to avoid failures in production, you should make sure that your production code will not fail when this change is applied in Notion. Once you have verified this, you can make the necessary change in your Notion database.

### ALL schema changes in your Notion databases will be reflected in your SDKs:

Next time you run the `generate` command – you will get an updated config file and a Client SDK, reflecting all the changes in the configuration of your Notion databases. If some property types or select/multi_select options have changed - you may also see some Typescript errors in your code. So simply fix those errors to adapt your code to the changes! 😎

---

License

MIT © Vels Lobak