Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/mrkmg/type-rest

A simple fetch wrapper made for TypeScript which allows developers to create and distribute a typed interface to their APIs.
https://github.com/mrkmg/type-rest

Last synced: 19 days ago
JSON representation

A simple fetch wrapper made for TypeScript which allows developers to create and distribute a typed interface to their APIs.

Awesome Lists containing this project

README

        

Type Rest
============

A simple fetch wrapper made for TypeScript which allows developers to
create and distribute a typed interface to their APIs.

## Quick Start

First, create an API instance and Spec:

api.ts
```typescript
import {typeRest, WithNone, WithBody, WithQuery, IHook} from "type-rest";

const loginHook: IHook = {
hook: (ev) => {
if (!ev.response.result) return Promise.reject(ev.response.error);
ev.instance._options.params.headers.token = ev.response.token;
},
method: "POST",
route: "/authentication/",
};

const logoutHook: IHook = {
hook: (ev) => {
delete ev.instance._options.params.headers.token;
},
method: "DELETE",
route: "/authentication/",
};

export const Api = typeRest("https://some-url/api", {
hooks: [loginHook, logoutHook],
params: {
mode: "cors",
}
});

export interface ApiRoute {
authentication: AuthenticationRoute;
todos: TodosRoute;
}

export interface AuthenticationRoute {
Get: WithNone<{authenticated: boolean}>;
Post: WithBody<{username: string, password: string}, {result: boolean, token?: string, error?: string}>;
Delete: WithNone;
}

export interface TodosRoute {
Get: WithQuery<{page: number, limit?: number}, Todo[]>;
Post: WithBody>, Todo>;
[todoId: number]: TodoRoute;
}

export interface TodoRoute {
Get: WithNone;
Patch: WithBody, Todo>;
Delete: WithNone;
}

export interface Todo {
id: number;
date: string;
title: string;
completed: boolean;
}
```

## Intended Use

Type Rest is intended to be used to provide a simple to use interface
to a well defined JSON-based API. All endpoints, query params, and body
params can be typed out via TypeScript and the provided types, and then
passed to an instance of type-rest which will then convert those typed
routes into callable functions.

Type Rest also provides a hook system to trigger actions before and
after a request. The hooks can modify requests as well as trigger
side-effects with the response.

In order to provide a typed interface to your api, you must first build
your API definition. This is done by using the included helper
definitions.

Under the hood, Type Rest uses fetch with one major difference. All
non-successful calls are rejected. This includes 4xx and 5xx server
errors.

## How to Use

Once you have a defined API and exported an instance of Type Rest with
your API definition, you or other developers, will be able to call
endpoints to your api.

From the example above:

test.ts
```typescript
import {Api} from "./api";

async function testApi() {
const authStatus = await Api.authentication.Get();

if (!authStatus.authenticated) {
await Api.authentication.Post({username: "test", password: "test"});
// Due to the hook define, this will add the appropriate token header to all future API calls
}

const todos = await Api.todos.Get({page: 2}); // List page two
const todo = await Api.todos[1].Get(); // Get to-do with ID 1
await Api.todos[1].Delete(); // Delete to-do with ID 1
}

```

## Building the API Definition

One of the core concepts of Type Rest is fully defining all the input
and output types of the restful calls. Using the powerful type system
built into TypeScript combined with the flexibility of Type Rest,
almost any situation can be accounted for.

There 4 concepts to creating your Typed API. First is the routes,
second are the end points, third are the input and output data, and finally
the instance. Routes are paths of your api, end points
are the HTTP verbs, data is the request and response data, and the
instance is what brings all those together into a usable api.

### Defining your api

Lets take the following API as an example:

```text
ROOT https://awesome-app/api/v1/

Authentication

GET /authentication
Get authentication status

POST /authentication
Authenticate with the api

DELETE /authentication
Remove authentication

GET /todos
Get a list of todos

GET /todos/{id}
Get a single todo

POST /todos
Create a todo

PATCH /todos/{id}
Update a todo

DELETE /todos/{id}
Delete a todo
```

Using the list of end points above, we can create all the definitions we need.

Looking at the api, we have two first-level routes: "authentication" and
"todos". Lets start with the authentication route.

Within authentication, we have 3 end-points, "GET, POST, and DELETE". We can
directly type those.

authentication-route.ts
```typescript
import {WithBody, WithNone} from "type-rest";

export interface IAuthenticationRoute {
Get: WithNone<{authenticated: boolean}>;
Post: WithBody<{username: string, password: string}, {result: boolean, token?: string, error?: string}>;
Delete: WithNone;
}
```

Next, we can start typing out the "todos" route. First thing to do is to type out a Todo.

todo.ts
```typescript
export interface ITodo {
id: number;
date: string;
title: string;
completed: boolean;
}
```

After we have the Todo interface, we will need to type out the route.

todos-route.ts
```typescript
import {WithBody, WithQuery} from "type-rest";
import {ITodo} from "./todo";

export interface ITodosRoute {
Get: WithQuery<{page: number, limit?: number}, ITodo[]>;
Post: WithBody>, ITodo>;
}
```

Here we see an advanced type. Basically what the `Pick>`
means is the all the keys of `Todo` except "id" and "completed".

But we are missing the routes which work on a single todo.
Lets type that route out separately first.

todo-route.ts
```typescript
import {WithBody, WithNone} from "type-rest";
import {ITodo} from "./todo";

export interface ITodoRoute {
Get: WithNone;
Patch: WithBody, ITodo>;
Delete: WithNone;
}
```

Now that we have that route, we can modify our "TodosRoute" to include
it. Seeing as the route is variable to the id of the Todo, we can use
the "index" property in typescript.

todos-route.ts
```typescript
import {WithBody, WithQuery} from "type-rest";
import {ITodo} from "./todo";
import {ITodoRoute} from "./todo-route";

export interface ITodosRoute {
Get: WithQuery<{page: number, limit?: number}, ITodo[]>;
Post: WithBody>, ITodo>;
[todoId: number]: ITodoRoute;
}
```

Now, all of our routes are defined. We just need to bring it all
together into a single "root" route which defines the entire API.

routes.ts
```typescript
import {IAuthenticationRoute} from "./authentication-route";
import {ITodosRoute} from "./todos-route";

export interface IAwesomeApiRoutes {
authentication: IAuthenticationRoute;
todos: ITodosRoute;
}
```

### Creating the instance

We have the entire API typed out, and want to be actually use it in
our application. This is very easy:

awesome-api.ts
```typescript
import {typeRest} from "type-rest";
import {IAwesomeApiRoutes} from "./routes";

export const AwesomeApi = typeRest("https://awesome-app/api/v1/");
```

Now, anywhere else in your app, you can use the "AwesomeApi" to make
requests to your api!

## Hooks

Built into type rest is the ability to hook into requests to modify
the api instance or trigger side effects. The primary purpose of the
hooks is to handle authentication tokens, headers, etc. Many API's
require a token to be sent with every request. This token can either
be given to the developer, or returned as part of an authentication request.

In the above example, we can see that the authentication result returns
a token. If we were required to pass this token in all future requests,
the user of the API would need to modify the api params.

```typescript
import {AwesomeApi} from "./awesome-api.ts"

const result = AwesomeApi.authentication.Post({username: "user", password: "pass"});

if (result.valid) {
AwesomeApi._options.params.headers["auth-token"] = result.token;
}
```

If you as the API developer wanted to automate this for the
consumers of your API, you can use the hooks system.

hooks.ts
```typescript
import {IHook} from "type-rest";

export const loginHook: IHook = {
hook: (ev) => {
if (!ev.response.result) { return Promise.reject(ev.response.error); }
ev.instance._options.params.headers["auth-token"] = ev.response.token;
},
method: "POST",
route: "/authentication/",
type: "post",
};

export const logoutHook: IHook = {
hook: (ev) => {
delete ev.instance._options.params.headers["auth-token"];
},
method: "DELETE",
route: "/authentication/",
type: "post",

};
```

awesome-api-with-hooks.ts
```typescript
import {typeRest} from "type-rest";
import {loginHook, logoutHook} from "./hooks";
import {IAwesomeApiRoutes} from "./routes";

export const AwesomeApi = typeRest("https://awesome-app/api/v1/", {
hooks: [loginHook, logoutHook],
});
```

In the above example, you can see two separate hooks, a login hook and
a logout hook. The login hook will only be executed on a "POST" call to
the /authentication/ route. the logout hook will only be executed on a
"DELETE" call the /authentication and will remove the token.

## Adding hooks at runtime

If you have an instance of Type Rest, and want to add a hook, you can
use the `_addHook` method. From the examples above:

```typescript
import {typeRest} from "type-rest";
import {loginHook, logoutHook} from "./hooks";
import {IAwesomeApiRoutes} from "./routes";

export const AwesomeApi = typeRest("https://awesome-app/api/v1/");

AwesomeApi._addHook(loginHook);
AwesomeApi._addHook(logoutHook);
```

Important: If `_addHook` is called from a sub-path, the hook will only apply
to the route and its child-routes, regardless of the "path" property.

## License

Copyright 2019 Kevin Gravier

Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files
(the "Software"), to deal in the Software without restriction,
including without limitation the rights to use, copy, modify, merge,
publish, distribute, sublicense, and/or sell copies of the Software, and
to permit persons to whom the Software is furnished to do so, subject
to the following conditions:

The above copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.