Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/kreteshq/retes

Typed, Declarative, Data-Driven & Functional Routing for Node.js (Express.js alternative in TypeScript)
https://github.com/kreteshq/retes

express nodejs routing

Last synced: about 1 month ago
JSON representation

Typed, Declarative, Data-Driven & Functional Routing for Node.js (Express.js alternative in TypeScript)

Awesome Lists containing this project

README

        

# Retes

Typed, Declarative, Data-Driven Routing for Node.js.

## What is Retes?

Retes is a routing library for Node.js written in TypeScript and inspired by Clojure's [Ring](https://github.com/ring-clojure/ring), [Compojure](https://github.com/weavejester/compojure) and [Retit](https://github.com/metosin/reitit). It is built directly on top of the Node.js `http` module, so you can use it as an alternative to Express or Koa.

- **Data-Driven:** In Retes you define routes using the existing data structures. This way, we limit the number of abstractions and we are able to easily transform and combine routes. Our routing description is declarative.
- **Typed:** The type system conveniently helps us control the shape of our routing
- **Battery-Included (wip):** Most common middlewares will be included out of the box

## Key Features

- built-in parsing of query params, body and route's dynamic segments
- built-in file uploading handling mechansim
- fast route matching (see [Benchmarks](#benchmarks))
- handlers are functions that take requests as input and return responses as output
- middlewares can be combined on per-route basis
- an HTTP response is just an object containing at least `statusCode` and `body` keys

## Why Retes?

- declarative route descriptions make them easily composable
- functional handlers are more natural fit for the HTTP flow
- common request/response transformations are already built-in
- typed routes make it easier to discover and control the shape of data flowing in and out

## Usage

Generate a new Retes app using `create-retes-app`. This will set up automatically the required project structure for you.

```
pnpm create retes-app my-retes-app
# or
yarn create retes-app my-retes-app
# or
npx create-retes-app@latest my-retes-app
```

Once the project is created, install its dependencies:

```
cd my-retes-app
```

```
pnpm install
# or
yarn
# or
npm install
```

Finally, start the application

```
pnpm start
```

The server application listens on the specified port, in our case `:3000`. Open [localhost:3000](http://localhost:3000) and test the routes.

## Features

### Params

Retes combines requests' query params, body params and segment params into `params`.

```ts
import { Response, Route, ServerApp } from 'retes';
import { Created } from 'retes/response';
import { GET, POST } from 'retes/route';

const routes = [
GET('/query-params', ({ params }) => OK(params)),
POST('/body-form', ({ params }) => OK(params)),
POST('/body-json', () => OK(params)),
GET('/segment/:a/:b', ({ params }) => OK(params)),
];

async function main() {
const app = new ServerApp(routes);
await app.start(3000);

console.log('started');
}

main();
```

This `GET` query

```
http :3000/query-params?a=1&b=2
```

returns

```http
HTTP/1.1 200 OK

{
"a": "1",
"b": "2"
}
```

This `POST` query with `Content-Type` set to `application/x-www-form-urlencoded; charset=utf-8`

```
http --form :3000/body-form a:=1 b:=2
```

returns

```http
HTTP/1.1 200 OK

{
"a": "1",
"b": "2"
}
```

This `POST` query with `Content-Type` set to `application/json`

```
http :3000/body-json a:=1 b:=2
```

returns

```http
HTTP/1.1 200 OK

{
"a": 1,
"b": 2
}
```

This `GET` request

```
http :3000/segment/1/2
```

returns

```http
HTTP/1.1 200 OK
{
"a": "1",
"b": "2"
}
```

### Convenience Wrappers for HTTP Responses

```ts
import { Response, Route, ServerApp } from 'retes';
import { Accepted, Created, InternalServerError, OK } from 'retes/response';
import { GET } from 'retes/route';

const routes = [
GET('/created', () => Created('payload')), // returns HTTP 201 Created
GET('/ok', () => OK('payload')), // returns HTTP 200 OK
GET('/accepted', () => Accepted('payload')), // returns HTTP 202 Accepted
GET('/internal-error', () => InternalServerError()), // returns HTTP 500 Internal Server Error
];

async function main() {
const app = new ServerApp(routes);
await app.start(3000);

console.log('started');
}

main();
```

### Middleware Composition on Per-Route Basis

```ts
import { ServerApp } from 'retes';
import { GET } from 'retes/route';

const { GET } = Route;

const prepend = next => request => `prepend - ${next()}`;
const append = next => request => `${next()} - append`;

const routes = [
GET('/middleware', () => 'Hello, Middlewares', {
middleware: [prepend, append],
}), // equivalent to: prepend(append(handler))
];

async function main() {
const app = new ServerApp(routes);
await app.start(3000);

console.log('started');
}

main();
```

## Benchmarks

WIP

## Roadmap

- [ ] infer types for dynamic segments from routes using the string literals feature from TypeScript 4.1 (in progress PR #1)