hapi-sequelize-crud ![CircleCI](https://circleci.com/gh/mdibaiee/hapi-sequelize-crud.svg?style=svg)
Automatically generate a RESTful API for your models and associations
This plugin depends on hapi-sequelize
.
npm install -S hapi-sequelize-crud
Configure
Please note that you should register hapi-sequelize-crud
after defining your
associations.
await register({
register: require('hapi-sequelize'),
options: { ... }
});
let db = server.plugins['hapi-sequelize'].db;
let models = db.sequelize.models;
associations(models);
await register({
register: require('hapi-sequelize-crud'),
options: {
prefix: '/v1',
name: 'db',
defaultConfig: { ... },
models: ['cat', 'dog']
models: [
{model: 'cat', methods: ['get', 'list']},
{model: 'dog'},
'cow',
{model: 'bat', methods: ['list'], config: { ... }},
{model: 'bat', methods: ['create']}
{model: 'fly', config: {
, ext: {
onPreHandler: (request, reply) => {
if (request.auth.hasAccessToFly) reply.continue()
else reply(Boom.unauthorized())
}
}
response: {
schema: {id: joi.string()},
modify: true
}
}}
]
}
});
Methods
- list: get all rows in a table
- get: get a single row
- scope: reference a sequelize scope
- create: create a new row
- destroy: delete a row
- destroyAll: delete all models in the table
- destroyScope: use a sequelize scope to find rows, then delete them
- update: update a row
where
queries
It's easy to restrict your requests using Sequelize's where
query option. Just pass a query parameter.
Team.findOne({ where: { city: 'windsor' }})
You can also do more complex queries by setting the value of a key to JSON.
Team.findOne({ where: { address: { city: 'windsor' }}})
include
queries
Getting related models is easy, just use a query parameter include
.
Team.findAll({include: City})
If you want to get multiple related models, just pass multiple include
parameters.
Team.findAll({include: [City, Uniform]})
For models that have a many-to-many relationship, you can also pass the plural version of the association.
Team.findAll({include: [Player]})
limit
and offset
queries
Restricting list (GET
) and scope queries to a restricted count can be done by passing limit=<number>
and/or offset=<number>
.
Team.findAll({limit: 10, offset: 10})
order
queries
You can change the order of the resulting query by passing order
to the query.
Team.findAll({order: ['name']})
Team.findAll({order: [['name', 'DESC']]})
Team.findAll({order: [['name'], ['city']]})
You can even order by associated models. Though there is a sequelize bug that might prevent this from working properly. A workaround is to &include
the model you're ordering by.
Player.findAll({order: [[{model: Team}, 'name']]})
Player.findAll({order: [[{model: Team}, 'name']], include: [Team]})
Authorization and other hooks
You can use Hapi's ext
option to interact with the request both before and after this module does. This is useful if you want to enforce authorization, or modify the request before or after this module does. Hapi has a full list of hooks you can use.
Modify the response format
By default, hapi-sequelize-crud
routes will respond with the full model. You can modify this using the built-in hapi settings.
await register({
register: require('hapi-sequelize-crud'),
options: {
…
{model: 'fly', config: {
response: {
schema: { id: joi.string() },
modify: true
}
}}
}
})
Full list of methods
Let's say you have a many-to-many
association like this:
Team.belongsToMany(Role, { through: 'TeamRoles' });
Role.belongsToMany(Team, { through: 'TeamRoles' });
You get these:
# get an array of records
GET /team/{id}/roles
GET /role/{id}/teams
# might also append `where` query parameters to search for
GET /role/{id}/teams?members=5
GET /role/{id}/teams?city=healdsburg
# you might also use scopes
GET /teams/{scope}/roles/{scope}
GET /team/{id}/roles/{scope}
GET /roles/{scope}/teams/{scope}
GET /roles/{id}/teams/{scope}
# get a single record
GET /team/{id}/role/{id}
GET /role/{id}/team/{id}
# create
POST /team/{id}/role
POST /role/{id}/team
# update
PUT /team/{id}/role/{id}
PUT /role/{id}/team/{id}
# delete
DELETE /team/{id}/roles #search and destroy
DELETE /role/{id}/teams?members=5
DELETE /team/{id}/role/{id}
DELETE /role/{id}/team/{id}
# include
# include nested associations (you can specify an array if includes)
GET /team/{id}/role/{id}?include=SomeRoleAssociation
# you also get routes to associate objects with each other
GET /associate/role/{id}/employee/{id} # associates role {id} with employee {id}
# you can specify a prefix to change the URLs like this:
GET /v1/team/{id}/roles