Expressive routing for HTTP service calls
This is an abstraction layer on top of http service calls e.g. ajax calls from the browser, or server-to-server http calls in node.
It's purpose is to expressively describe, validate, and hook into the various API calls an application might make.
Features and use cases:
- Easily create new API service calls with less repeated code (like copy/pasting the same
request
module calls everywhere) - Event hooks, which can be used for centralized logging for every service call made
- Automatic validation of input params as well as responses
- Easily swap out the underlying network library, e.g. swapping out
fetch
foraxios
- Mocking APIs (see express.js mocking example)
npm install scenic-route-client ajv qs --save
We will create a router to the Github API, using the request
module to facilitate the HTTP calls.
const request = require('request');
const { createRouter } = require('scenic-route-client');
const router = createRouter('https://api.github.com', (url, method, input, callback) => {
request({ url, method,headers: input.headers, qs: input.query, body: input.body, json: true }, (err, res, payload) => {
callback(err, payload, res.statusCode, res.headers);
});
});
We will then define two service calls:
GET https://api.github.com/search/users
GET https://api.github.com/search/repositories}
router.group('/search', (router) => {
router.get('/users', 'searchUsers');
router.get('/repositories', 'searchRepos');
});
We then send the following requests:
GET https://api.github.com/search/users?q=shaunpersad
GET https://api.github.com/search/repositories?q=scenic-route-client
const searchUsers = router.operation('searchUsers');
const searchRepos = router.operation('searchRepos');
searchUsers({ query: { q: 'shaunpersad' } }, (err, payload) => {
// payload is the result of the API call
});
searchRepos({ query: { q: 'scenic-route-client' } }, (err, payload) => {
// payload is the result of the API call
});
We could make the route definitions even more explict by defining their accepted parameters:
GET https://api.github.com/search/users?q={string}
GET https://api.github.com/search/repositories?q={string}
router.group({
prefix: '/search',
inputProperties: {
query: {
q: {
type: 'string'
}
}
}
}, (router) => {
router.get('/users', 'searchUsers');
router.get('/repositories', 'searchRepos');
});
We could also define what we expect back:
router.group({
prefix: '/search',
inputProperties: {
query: {
q: {
type: 'string'
}
}
},
success: {
'200': {
type: 'object',
properties: {
items: {
type: 'array',
items: {
type: 'object',
properties: {
id: {
type: 'number'
}
},
required: ['id']
}
}
},
required: ['items']
}
}
}, (router) => {
router.get('/users', 'searchUsers');
router.get('/repositories', 'searchRepos');
});
The above definitions will then automatically validate both the requests and the responses sent.