Create a REST API client through a declarative API definition, based on io-ts type definitions. Integrates with state management libraries, like redux.
Example:
import { RestApi, createAgent } from '@mkrause/lifecycle-rest';
import * as t from 'io-ts';
// Define your data types
const User = t.type({ name: t.string });
const UsersCollection = t.array(User);
// Create an HTTP agent (axios)
const agent = createAgent({
baseURL: 'https://example.com/api',
});
const api = RestApi({ agent }, RestApi.Item(t.unknown, {
resources: {
users: RestApi.Collection(UsersCollection, {
uri: 'users',
// Custom methods
methods: {
search: RestApi.decorateMethod(async ({ agent, uri }, query) => {
return await agent.get(uri, query);
}),
},
entry: RestApi.Item(User),
}),
},
}));
// Call the API directly
const users = await api.users.list(); // GET /api/users
// Index into a collection to access an entry (configurable through the `entry` property)
const john = await api.users('john').get(); // GET /api/users/john
// Or, dispatch to a redux store in order to store the result
dispatch(api.users.search({ name: 'Alice' })); // GET /api/users?name=Alice
This library exports a RestApi
function, which you can use to define your API. It takes two arguments: the agent which is used to make the HTTP requests, and an API definition that declares the various kinds of resources that your REST API is made out of.
import RestApi from '@mkrause/lifecycle-rest';
const api = RestApi({ agent: <agent> }, <api-definition>);
The agent should be an instance of the axios library. We provide a createAgent
helper that you can use that comes with a few useful defaults.
import RestApi, { createAgent } from '@mkrause/lifecycle-rest';
const agent = createAgent({
baseURL: 'https://example.com/api',
});
The API definition consists of a tree of resource definitions. A resource definition describes some REST resource in your API. For example, you might have an endpoint hello
that takes a name and returns a greeting. You could define that resource as follows:
const greetingApi = RestApi({ agent }, RestApi.Item(t.unknown, {
uri: 'hello',
}));
// GET https://example.com/api/hello?name=Bob
const greeting = await greetingApi.get({ name: 'Bob' }); // Returns "Hello Bob!"
Each resource may have subresources. Subresources are accessed as properties on the resulting API client:
const api = RestApi({ agent }, RestApi.Item(t.unknown, {
resources: {
users: RestApi.Collection(UsersCollection),
},
}));
// GET https://example.com/api/users
const users = await api.users.list();
Notice that we've defined the users
subresource as a Collection
resource. There are several types of resources available, and each comes with their own methods.
RestApi.Item
RestApi.Collection
Note that if you do not specify the type of the resource explicitly (as in the "greeting" example), then we will create an Item resource by default.
Properties of subresources (like the uri
) are relative to their parent resource by default. If the uri
is not specified we will use the name of the subresource (e.g. users
in the example above).
Configuration:
uri
: The URI of the resource. If relative, will be created relative to the URI of the parent resource. Defaults to the name of the subresource (or empty""
if this is the root resource).methods
: Custom methods definitions (see below).resources
: A map of subresources, if any.
Custom methods can be defined as follows:
const api = RestApi({ agent }, RestApi.Item(t.unknown, {
methods: {
getCustom: RestApi.decorateMethod(async ({ agent, uri }, ...args) => {
// Here, the first argument is the resource definition, and `args` contains any remaining arguments
return await agent.get(uri);
}),
},
}));
api.getCustom('foo');
RestApi.Item(schema, resourceSpec)
There are a number of methods implemented on this resource by default:
get(params : object)
: Perform a GET request, usingparams
as the query parameters.put(item : Item, params : object)
: Perform a PUT request, whereitem
is the resource to send.patch(item : Item, params : object)
: Perform a PUT request, whereitem
is the resource to send.delete(item : Item, params : object)
: Perform a DELETE request, whereitem
is the resource to delete.post(body : unknown, params : object)
: Perform a generic POST request (no decoding performed).
RestApi.Collection(CollectionSchema, resourceSpec)
Configuration:
entry
: Resource definition for entries of this collection.
There are a number of methods implemented on this resource by default:
get(params : object)
: Perform a GET request, usingparams
as the query parameters.list(params : object)
: (Alias forget
.)put(collection : Collection, params : object)
: Perform a PUT request, wherecollection
is the resource to send.create(entry : Entry, params : object)
. Create a new entry in the collection. Requires theentry
property to be defined in order to determine the resource type (Entry
).post(body : unknown, params : object)
: Perform a generic POST request (no decoding performed).
lifecycle-rest
comes with integration with redux out of the box. To use it, you will need to install the middleware in order to be able to dispatch REST API calls.
import { createStore, applyMiddleware } from 'redux';
import { redux as lifecycleRedux } from '@mkrause/lifecycle-rest';
const lifecycleMiddleware = lifecycleRedux.middleware();
const store = createStore(reducer, initialState, applyMiddleware(lifecycleMiddleware));
Now, you can use dispatch
to dispatch API calls as follows:
const api = RestApi(...);
// Somewhere in your application:
dispatch(api.users.get());
This will result in two actions being dispatched: a loading action right at the start of the call. Then later either a ready or failed action when the API call gets a response. You can handle these actions yourself in your reducer, or if you want you can use the standard reducer provided by this library:
import { createStore, applyMiddleware } from 'redux';
import { redux as lifecycleRedux } from '@mkrause/lifecycle-rest';
const lifecycleMiddleware = lifecycleRedux.middleware();
const reducers = [lifecycleRedux.reducer]; // Add your own reducers
const reducer = (state, action) =>
reducers.reduce((state, reducer) => reducer(state, action), state);
const store = createStore(reducer, initialState, applyMiddleware(lifecycleMiddleware));
There are plenty of libraries already out there to create REST clients, see here for some popular examples. The main reason I've created this library is because I needed it to integrate with a set of state management libraries I've called lifecycle.
But there are a few other reasons I think makes this library stand out from the rest (no pun intended!):
-
Upfront, declarative definition as opposed to defining endpoints dynamically. By teaching the client about the API resources up front we can provide smarter, more error-proof handling of API requests.
-
API calls don't just return data, but also provide descriptions of the state updates that need to be done to incorporate this data in your state tree (redux, or other state management libraries). For example, depending on the type of API request performed, the state update may be partial or full, and this is reflected through the
status
of the resulting state item.