Cloudlfare Workers Router
This is a simple & lightweight router designed for Cloudflare Workers.
Its main goal is to provide developers with a simple & comfortable interface to
develop a multi-route worker application.
Internally it uses a combination of two radix trees to build routes, allowing the developer
to add regex segments as well as parameter segments to the routes.
tl;dr
const { Router, App } = require('@control/cloudflare-workers-router');
const router = new Router();
router.get('/resources', (request, params={}, pre_processed=null, fetch_event) => {
return [{
id: 1
}, {
id: 2
}, {
id: 3
},]
});
router.any('/resource/1', (request, params={}, pre_processed=null, fetch_event) => {
return {
id: 5,
next: '/resource/2'
}
});
const app = new App(router);
app.pre_process((pre_processed_aggregator, request, params, fetch_event) => {
return request.json().then(json => json.data);
});
app.pre_process((pre_processed_aggregator, request, params, fetch_event) => {
return pre_processed_aggregator.map(data => do_something(data));
});
app.pre_process((pre_processed_aggregator, request, params, fetch_event) => {
const auth = request.headers.get('Authorization');
if(auth !== 'my_super_secret_password' && params.object_id === '5') {
return new Response(null, {
status: 403
});
}
return pre_processed_aggregator;
});
app.post_process((response, request, params, event) => {
const headers = {
'Access-Control-Allow-Origin': request.headers.get('origin'),
'Access-Control-Allow-Methods': 'GET,OPTIONS,POST',
'Access-Control-Max-Age': '86400'
};
for(const [ key, value ] of Object.entries(headers)) {
response.headers.set(key, value);
}
return response;
});
app.listen();
Usage
The module is divided into two main objects, Routers
and the App
.
The App
module is a small placeholder to allow you to add pre-processing and post-processing to your app, but it is not really needed.
You'll mostly use the routers to add your logic.
This section will start with the Routers so you get an overview, and then we will show the app
so you can see how it is implemented.
Router
Using this library is extremely simple, and if you've ever used any other http framekwork on Nodejs, you'll find
it quite similar.
Look at this ping pong example router:
const { Router } = require('@control/cloudflare-workers-router');
const sub_router = new Router();
sub_router.get('/resource', (request) => {
return {
id: 1
}
});
const router = new Router();
router.get('/ping', (request) => {
return 'pong';
});
router.use('/sub', sub_router);
module.exports = router;
This router will register the
/ping
route, which will simply respond with pong
- all the
sub_router
routes, with the /sub
segment prefixing all of those routes (/sub/resource
)
Please note that when using sub-routers they must be complete before merging them.
Subtrees are merged into the main tree, which means that modifying them later will have no effect on the main tree.
You can call any of the following methods to register routes:
Method | HTTP Verb |
---|
any | Any of the following |
get | GET |
post | POST |
put | PUT |
patch | PATCH |
delete | DELETE |
head | HEAD |
options | OPTIONS |
Sub-routers
Sub routers are there to help you divide your app in different files and make your code more readable.
Otherwise they don't have any other purpose (especially since this lib doesn't support middlewares).
As stated above, sub-routers are merged into the main tree (of the router calling .use()
), thus making it impossible
for you to make post-merge changes and have them reflected on your final entrypoint.
Not using app
If you want to skip the app
module, you can use the Router.run
method, it takes the request and the pre-processed params, resolves and executes the route, and formats the response.
const router = new Router();
router.get('ping', () => 'pong');
addEventListener('fetch', event => {
event.respondWith(
router.run(event.request, null)
);
});
App
The App module has two main methods, pre_process
and post_process
.
These methods register callbacks to run
- preprocess: before your logic but after route resolution has been made, allowing you to access the route params.
- postprocess: after your logic AND after response formatting which means that you get a
Response
with a JSON-strignified body (if the result was not already a string, more on this later)
These methods allow you to add custom logic for authentication or post-processing needs, like CORS header appending.
Response formatting
This module will pre-format the response for you, allowing you to "just" return whatever you need to send to your client.
However, this module supposes that you will be sending a string or a json formatted blob, which means that if you return any of the following objects they will be JSON-stringified:
Types |
---|
Object ({} , [] , or other) |
true or false |
Any number recognized by !Number.isNaN |