Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/tsndr/cloudflare-worker-router
A super lightweight router (1.0K) with middleware support and ZERO dependencies for Cloudflare Workers.
https://github.com/tsndr/cloudflare-worker-router
api cloudflare cloudflare-worker cloudflare-workers framework middleware router routing worker workers
Last synced: 3 days ago
JSON representation
A super lightweight router (1.0K) with middleware support and ZERO dependencies for Cloudflare Workers.
- Host: GitHub
- URL: https://github.com/tsndr/cloudflare-worker-router
- Owner: tsndr
- License: mit
- Created: 2021-02-13T00:22:34.000Z (almost 4 years ago)
- Default Branch: main
- Last Pushed: 2024-10-26T21:08:32.000Z (3 months ago)
- Last Synced: 2025-01-10T21:05:28.491Z (11 days ago)
- Topics: api, cloudflare, cloudflare-worker, cloudflare-workers, framework, middleware, router, routing, worker, workers
- Language: TypeScript
- Homepage:
- Size: 534 KB
- Stars: 232
- Watchers: 6
- Forks: 29
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- Funding: .github/FUNDING.yml
- License: LICENSE
Awesome Lists containing this project
README
# Cloudflare Workers Router
Cloudflare Workers Router is a super lightweight router (1.0K gzipped) with middleware support and **ZERO dependencies** for [Cloudflare Workers](https://workers.cloudflare.com/).
When I was trying out Cloudflare Workers I almost immediately noticed how fast it was compared to other serverless offerings. So I wanted to build a full-fledged API to see how it performs doing real work, but since I wasn't able to find a router that suited my needs I created my own.
## Contents
- [Features](#features)
- [Usage](#usage)
- [Reference](#reference)
- [Getting started](#getting-started)## Features
- ZERO dependencies
- Lightweight (1.0K gzipped)
- Fully written in TypeScript
- Built specifically around Middlewares
- Debug-Mode, CORS and Bearer helpers
- Allows accessing request body multiple times## Usage
Migrating from `v2.x.x`, check out the [Migration Guide](MIGRATION.md).
### TypeScript Example
```typescript
import { Router } from '@tsndr/cloudflare-worker-router'// Env Types
export type Var = T
export type Secret = Texport type Env = {
// Example binding to KV. Learn more at https://developers.cloudflare.com/workers/runtime-apis/kv/
// MY_KV_NAMESPACE: KVNamespace
//
// Example binding to Durable Object. Learn more at https://developers.cloudflare.com/workers/runtime-apis/durable-objects/
// MY_DURABLE_OBJECT: DurableObjectNamespace
//
// Example binding to R2. Learn more at https://developers.cloudflare.com/workers/runtime-apis/r2/
// MY_BUCKET: R2BucketENVIRONMENT: Var<'dev' | 'prod'>
SECRET_TOKEN: Secret
}// Request Extension
export type ExtReq = {
userId?: number
}// Context Extension
export type ExtCtx = {
//sentry?: Toucan
}// Initialize Router
const router = new Router()// Enabling build in CORS support
router.cors()// Register global middleware
router.use(({ env, req }) => {
// Intercept if token doesn't match
if (req.headers.get('authorization') !== env.SECRET_TOKEN)
return new Response(null, { status: 401 })
})// Simple get
router.get('/user', () => {
return Response.json({
id: 1,
name: 'John Doe'
})
})// Post route with url parameter
router.post('/user/:id', ({ req }) => {const userId = req.params.id
// Do stuff
if (!true) {
return Response.json({
error: 'Error doing stuff!'
}, { status: 400 })
}return Response.json({ userId }, { status: 204 })
})// Delete route using a middleware
router.delete('/user/:id', ({ env, req }) => {
if (req.headers.get('authorization') === env.SECRET_TOKEN)
return new Response(null, { status: 401 })}, ({ req }) => {
const userId = req.params.id
// Do stuff...
return Response.json({ userId })
})// Listen Cloudflare Workers Fetch Event
export default {
async fetch(request: Request, env: Env, ctx: ExecutionContext) {
return router.handle(request, env, ctx)
}
}```
JavaScript Example
```javascript
import { Router } from '@tsndr/cloudflare-worker-router'// Initialize router
const router = new Router()// Enabling build in CORS support
router.cors()// Register global middleware
router.use(({ env, req }) => {
// Intercept if token doesn't match
if (req.headers.get('authorization') !== env.SECRET_TOKEN)
return new Response(null, { status: 401 })
})// Simple get
router.get('/user', () => {
return Response.json({
id: 1,
name: 'John Doe'
})
})// Post route with url parameter
router.post('/user/:id', ({ req }) => {const userId = req.params.id
// Do stuff
if (!true) {
return Response.json({
error: 'Error doing stuff!'
}, { status: 400 })
}return Response.json({ userId }, { status: 204 })
})// Delete route using a middleware
router.delete('/user/:id', ({ env, req }) => {
if (req.headers.get('authorization') === env.SECRET_TOKEN)
return new Response(null, { status: 401 })}, ({ req }) => {
const userId = req.params.id
// Do stuff...
return Response.json({ userId })
})// Listen Cloudflare Workers Fetch Event
export default {
async fetch(request, env, ctx) {
return router.handle(request, env, ctx)
}
}
```## Reference
### `router.debug([state = true])`
Enable or disable debug mode. Which will return the `error.stack` in case of an exception instead of and empty `500` response. Debug mode is disabled by default.
#### `state`
State is a `boolean` which determines if debug mode should be enabled or not (default: `true`)Key | Type | Default Value
---------------------- | --------- | -------------
`state` | `boolean` | `true`### `router.use([...handlers])`
Register a global middleware handler.
#### `handler(ctx)`
Handler is a `function` which will be called for every request.
#### `ctx`
Object containing `env`, [`req`](#req-object)
### `router.cors([config])`
If enabled will overwrite other `OPTIONS` requests.
#### `config` (object, optional)
Key | Type | Default Value
-------------------------- | ---------- | -------------
`allowOrigin` | `string` | `*`
`allowMethods` | `string` | `*`
`allowHeaders` | `string` | `*`
`allowCredentials` | `boolean` | `undefined`
`vary` | `string` | `undefined`
`maxAge` | `integer` | `86400`
`optionsSuccessStatus` | `integer` | `204`### Supported Methods
- `router.any(url, [...handlers])`
- `router.delete(url, [...handlers])`
- `router.get(url, [...handlers])`
- `router.head(url, [...handlers])`
- `router.options(url, [...handlers])`
- `router.patch(url, [...handlers])`
- `router.post(url, [...handlers])`
- `router.put(url, [...handlers])`#### `url` (string)
The URL starting with a `/`.
Supports the use of dynamic parameters, prefixed with a `:` (i.e. `/user/:userId/edit`) which will be available through the [`req`-Object](#req-object) (i.e. `req.params.userId`).#### `handlers` (function, optional)
An unlimited number of functions getting [`ctx`](#ctx-object) passed into them.
### `ctx`-Object
Key | Type | Description
--------- | ------------------- | -----------
`env` | `object` | Environment
`req` | `req`-Object | Request Object
`ctx` | `ctx`-Object | Cloudflare's `ctx`-Object### `req`-Object
Key | Type | Description
--------- | ------------------- | -----------
`body` | `object` / `string` | Only available if method is `POST`, `PUT`, `PATCH` or `DELETE`. Contains either the received body string or a parsed object if valid JSON was sent.
`headers` | `Headers` | Request [Headers Object](https://developer.mozilla.org/en-US/docs/Web/API/Headers)
`method` | `string` | HTTP request method
`params` | `object` | Object containing all parameters defined in the url string
`query` | `object` | Object containing all query parameters## Getting started
Please follow Cloudflare's [Get started guide](https://developers.cloudflare.com/workers/get-started/guide/) to install wrangler.
#### Initialize Project
```bash
wrangler init
```Use of TypeScript is strongly encouraged :)
```bash
npm i @tsndr/cloudflare-worker-router
```### TypeScript (
src/index.ts
)```typescript
import { Router } from '@tsndr/cloudflare-worker-router'// Env Types
export type Var = T
export type Secret = Texport type Env = {
// Example binding to KV. Learn more at https://developers.cloudflare.com/workers/runtime-apis/kv/
// MY_KV_NAMESPACE: KVNamespace
//
// Example binding to Durable Object. Learn more at https://developers.cloudflare.com/workers/runtime-apis/durable-objects/
// MY_DURABLE_OBJECT: DurableObjectNamespace
//
// Example binding to R2. Learn more at https://developers.cloudflare.com/workers/runtime-apis/r2/
// MY_BUCKET: R2Bucket
//
// Example Variable
// ENVIRONMENT: Var<'dev' | 'prod'>
//
// Example Secret
// JWT_SECRET: Secret
}// Request Extension
export type ExtReq = {
userId?: number
}// Context Extension
export type ExtCtx = {
//sentry?: Toucan
}// Handler Type
export type Handler = RouterHandler// Initialize Router
const router = new Router()// Enable Debug Mode
router.debug()// Enabling build in CORS support
//router.cors()/// Example Route
//
// router.get('/hi', async () => {
// return new Response('Hello World')
// })/// Example Route for splitting into multiple files
//
// const helloHandler: Handler = async () => {
// return new Response('Hello World')
// }
//
// router.get('/hellow', helloHandler)// TODO: add your routes here
export default {
async fetch(request: Request, env: Env, ctx: ExecutionContext) {
return router.handle(request, env, ctx)
}
}
```### JavaScript (
src/index.js
)Consider using TypeScript instead :)
```javascript
import { Router } from '@tsndr/cloudflare-worker-router'const router = new Router()
// Enable Debug Mode
//router.debug()// Enabling build in CORS support
//router.cors()/// Example Route
//
// router.get('/hi', async () => {
// return new Response('Hello World')
//})/// Example Route for splitting into multiple files
//
// async function hiHandler() {
// return new Response('Hello World')
// }
//
// router.get('/hi', hiHandler)// TODO: add your routes here
export default {
async fetch(request, env, ctx) {
return router.handle(request, env, ctx)
}
}
```