Compare commits

...

13 Commits

Author SHA1 Message Date
Joey Baker
facde8d542
Merge pull request #39 from rithalnz/patch-1
Create LICENSE
2021-03-13 11:35:24 -08:00
rithalnz
0cbc28ef90
Create LICENSE
Added MIT License
2021-03-10 12:49:07 -08:00
Mahdi Dibaiee
a6590a9650 fix(crud): see 83eadf0929a2506541e34b4a6f73d46bfa2404d3 and #37 2017-02-04 22:18:22 +03:30
Mahdi Dibaiee
85b52fd5ab fix(crud): should ignore JoinTable models while creating the routes -
solves #37
2017-02-04 22:11:58 +03:30
Joey Baker
5ba9d7d261 2.9.1 2016-11-01 19:10:04 -07:00
Joey Baker
07837ef36c Merge pull request #33 from mdibaiee/add-limit-offset-tests
Add integration tests for limit/offset
2016-11-01 19:09:45 -07:00
Joey Baker
25501bbb10 Test add integration tests for limit/offset
#28
2016-11-01 17:39:35 -07:00
Joey Baker
a335471f02 Fix(crud/list) 404 on no results 2016-11-01 17:39:16 -07:00
Joey Baker
ce26814f74 Test add a returnsAll scope to player fixture 2016-11-01 17:38:55 -07:00
Joey Baker
d1fc6d46e8 2.9.0 2016-10-31 14:24:12 -07:00
Joey Baker
4e94c7f825 Merge pull request #32 from mdibaiee/more-order-integration
Feat ordering by associated models now works
2016-10-31 14:23:43 -07:00
Joey Baker
c289fb2ed4 Feat ordering by associated models now works
It's now possible to order by associated models. This technically might
have worked before b/c we were parsing JSON sent to `order`, but I'm
pretty sure it wouldn't actually work b/c we never grabbed the actual
model to associate by. Regardless, this actually enables things and adds
tests to prove it.

Note: there's a sequelize bug that's poorly reported but definitely
known where `order` with associated models can fail because the sql
generated doesn't include a join. So, I added docs noting that and a
`test.failing` so that we'll be notified when that bug is fixed and can
remove the note.
2016-10-31 12:48:34 -07:00
Joey Baker
e1b851f932 Test: add a second mock team
So that we can test complex sorts
2016-10-31 12:41:34 -07:00
11 changed files with 261 additions and 36 deletions

21
LICENSE Normal file
View File

@ -0,0 +1,21 @@
The MIT License (MIT)
Copyright (c) 2021 The hapi-sequelize-crud Authors
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.

View File

@ -172,6 +172,22 @@ Team.findAll({order: [['name', 'DESC']]})
Team.findAll({order: [['name'], ['city']]}) Team.findAll({order: [['name'], ['city']]})
``` ```
You can even order by associated models. Though there is a [sequelize bug](https://github.com/sequelize/sequelize/issues?utf8=%E2%9C%93&q=is%3Aissue%20is%3Aopen%20order%20join%20) that might prevent this from working properly. A workaround is to `&include` the model you're ordering by.
```js
// returns the players ordered by the team name
// GET /players?order[0]={"model": "Team"}&order[0]=name
// results in a Sequelize query:
Player.findAll({order: [[{model: Team}, 'name']]})
// if the above returns a Sequelize error: `No such column Team.name`,
// you can work around this by forcing the join into the query:
// GET /players?order[0]={"model": "Team"}&order[0]=name&include=team
// results in a Sequelize query:
Player.findAll({order: [[{model: Team}, 'name']], include: [Team]})
```
## Authorization and other hooks ## Authorization and other hooks
You can use Hapi's [`ext` option](http://hapijs.com/api#route-options) 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](http://hapijs.com/api#request-lifecycle) you can use. You can use Hapi's [`ext` option](http://hapijs.com/api#route-options) 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](http://hapijs.com/api#request-lifecycle) you can use.

View File

@ -1,6 +1,6 @@
{ {
"name": "hapi-sequelize-crud", "name": "hapi-sequelize-crud",
"version": "2.8.0", "version": "2.9.3",
"description": "Hapi plugin that automatically generates RESTful API for CRUD", "description": "Hapi plugin that automatically generates RESTful API for CRUD",
"main": "build/index.js", "main": "build/index.js",
"config": { "config": {

View File

@ -0,0 +1,94 @@
import test from 'ava';
import 'sinon-bluebird';
import setup from '../test/integration-setup.js';
const STATUS_OK = 200;
const STATUS_NOT_FOUND = 404;
setup(test);
test('/players?limit=2', async (t) => {
const { server } = t.context;
const limit = 2;
const url = `/players?limit=${limit}`;
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
t.is(result.length, limit);
});
test('/players?limit=2&offset=1', async (t) => {
const { server } = t.context;
const limit = 2;
const url = `/players?limit=${limit}&offset=1`;
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
t.is(result.length, limit);
});
test('/players?limit=2&offset=2', async (t) => {
const { server } = t.context;
const limit = 2;
const url = `/players?limit=${limit}&offset=2`;
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
t.is(result.length, 1, 'with only 3 players, only get 1 back with an offset of 2');
});
test('/players?limit=2&offset=20', async (t) => {
const { server } = t.context;
const limit = 2;
const url = `/players?limit=${limit}&offset=20`;
const method = 'GET';
const { statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_NOT_FOUND, 'with a offset/limit greater than the data, returns a 404');
});
test('scope /players/returnsAll?limit=2', async (t) => {
const { server } = t.context;
const limit = 2;
const url = `/players/returnsAll?limit=${limit}`;
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
t.is(result.length, limit);
});
test('scope /players/returnsAll?limit=2&offset=1', async (t) => {
const { server } = t.context;
const limit = 2;
const url = `/players/returnsAll?limit=${limit}&offset=1`;
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
t.is(result.length, limit);
});
test('scope /players/returnsAll?limit=2&offset=2', async (t) => {
const { server } = t.context;
const limit = 2;
const url = `/players/returnsAll?limit=${limit}&offset=2`;
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
t.is(result.length, 1, 'with only 3 players, only get 1 back with an offset of 2');
});
test('scope /players/returnsAll?limit=2&offset=20', async (t) => {
const { server } = t.context;
const limit = 2;
const url = `/players/returnsAll?limit=${limit}&offset=20`;
const method = 'GET';
const { statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_NOT_FOUND, 'with a offset/limit greater than the data, returns a 404');
});

View File

@ -9,7 +9,7 @@ setup(test);
test('/players?order=name', async (t) => { test('/players?order=name', async (t) => {
const { server, instances } = t.context; const { server, instances } = t.context;
const { player1, player2 } = instances; const { player1, player2, player3 } = instances;
const url = '/players?order=name'; const url = '/players?order=name';
const method = 'GET'; const method = 'GET';
@ -18,11 +18,12 @@ test('/players?order=name', async (t) => {
// this is the order we'd expect the names to be in // this is the order we'd expect the names to be in
t.is(result[0].name, player1.name); t.is(result[0].name, player1.name);
t.is(result[1].name, player2.name); t.is(result[1].name, player2.name);
t.is(result[2].name, player3.name);
}); });
test('/players?order=name%20ASC', async (t) => { test('/players?order=name%20ASC', async (t) => {
const { server, instances } = t.context; const { server, instances } = t.context;
const { player1, player2 } = instances; const { player1, player2, player3 } = instances;
const url = '/players?order=name%20ASC'; const url = '/players?order=name%20ASC';
const method = 'GET'; const method = 'GET';
@ -31,24 +32,26 @@ test('/players?order=name%20ASC', async (t) => {
// this is the order we'd expect the names to be in // this is the order we'd expect the names to be in
t.is(result[0].name, player1.name); t.is(result[0].name, player1.name);
t.is(result[1].name, player2.name); t.is(result[1].name, player2.name);
t.is(result[2].name, player3.name);
}); });
test('/players?order=name%20DESC', async (t) => { test('/players?order=name%20DESC', async (t) => {
const { server, instances } = t.context; const { server, instances } = t.context;
const { player1, player2 } = instances; const { player1, player2, player3 } = instances;
const url = '/players?order=name%20DESC'; const url = '/players?order=name%20DESC';
const method = 'GET'; const method = 'GET';
const { result, statusCode } = await server.inject({ url, method }); const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK); t.is(statusCode, STATUS_OK);
// this is the order we'd expect the names to be in // this is the order we'd expect the names to be in
t.is(result[0].name, player2.name); t.is(result[0].name, player3.name);
t.is(result[1].name, player1.name); t.is(result[1].name, player2.name);
t.is(result[2].name, player1.name);
}); });
test('/players?order[]=name', async (t) => { test('/players?order[]=name', async (t) => {
const { server, instances } = t.context; const { server, instances } = t.context;
const { player1, player2 } = instances; const { player1, player2, player3 } = instances;
const url = '/players?order[]=name'; const url = '/players?order[]=name';
const method = 'GET'; const method = 'GET';
@ -57,19 +60,74 @@ test('/players?order[]=name', async (t) => {
// this is the order we'd expect the names to be in // this is the order we'd expect the names to be in
t.is(result[0].name, player1.name); t.is(result[0].name, player1.name);
t.is(result[1].name, player2.name); t.is(result[1].name, player2.name);
t.is(result[2].name, player3.name);
}); });
test('/players?order[0]=name&order[0]=DESC', async (t) => { test('/players?order[0]=name&order[0]=DESC', async (t) => {
const { server, instances } = t.context; const { server, instances } = t.context;
const { player1, player2 } = instances; const { player1, player2, player3 } = instances;
const url = '/players?order[0]=name&order[0]=DESC'; const url = '/players?order[0]=name&order[0]=DESC';
const method = 'GET'; const method = 'GET';
const { result, statusCode } = await server.inject({ url, method }); const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK); t.is(statusCode, STATUS_OK);
// this is the order we'd expect the names to be in // this is the order we'd expect the names to be in
t.is(result[0].name, player2.name); t.is(result[0].name, player3.name);
t.is(result[1].name, player2.name);
t.is(result[2].name, player1.name);
});
// multiple sorts
test('/players?order[0]=active&order[0]=DESC&order[1]=name&order[1]=DESC', async (t) => {
const { server, instances } = t.context;
const { player1, player2, player3 } = instances;
const url = '/players?order[0]=name&order[0]=DESC&order[1]=teamId&order[1]=DESC';
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
// this is the order we'd expect the names to be in
t.is(result[0].name, player3.name);
t.is(result[1].name, player2.name);
t.is(result[2].name, player1.name);
});
// this will fail b/c sequelize doesn't correctly do the join when you pass
// an order. There are many issues for this:
// eslint-disable-next-line
// https://github.com/sequelize/sequelize/issues?utf8=%E2%9C%93&q=is%3Aissue%20is%3Aopen%20order%20join%20
//
// https://github.com/sequelize/sequelize/issues/5353 is a good example
// if this test passes, that's great! Just remove the workaround note in the
// docs
// eslint-disable-next-line
test.failing('sequelize bug /players?order[0]={"model":"Team"}&order[0]=name&order[0]=DESC', async (t) => {
const { server, instances } = t.context;
const { player1, player2, player3 } = instances;
const url = '/players?order[0]={"model":"Team"}&order[0]=name&order[0]=DESC';
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
// this is the order we'd expect the names to be in
t.is(result[0].name, player3.name);
t.is(result[1].name, player1.name); t.is(result[1].name, player1.name);
t.is(result[2].name, player2.name);
});
// b/c the above fails, this is a work-around
test('/players?order[0]={"model":"Team"}&order[0]=name&order[0]=DESC&include=team', async (t) => {
const { server, instances } = t.context;
const { player1, player2, player3 } = instances;
const url = '/players?order[0]={"model":"Team"}&order[0]=name&order[0]=DESC&include=team';
const method = 'GET';
const { result, statusCode } = await server.inject({ url, method });
t.is(statusCode, STATUS_OK);
// this is the order we'd expect the names to be in
t.is(result[0].name, player3.name);
t.is(result[1].name, player1.name);
t.is(result[2].name, player2.name);
}); });
test('invalid column /players?order[0]=invalid', async (t) => { test('invalid column /players?order[0]=invalid', async (t) => {

View File

@ -165,6 +165,8 @@ export const list = ({ server, model, prefix = '/', config }) => {
where, include, limit, offset, order, where, include, limit, offset, order,
}); });
if (!list.length) return void reply(notFound('Nothing found.'));
reply(list.map((item) => item.toJSON())); reply(list.map((item) => item.toJSON()));
}, },

View File

@ -38,7 +38,6 @@ const register = (server, options = {}, next) => {
// Join tables // Join tables
if (model.options.name.singular !== model.name) continue; if (model.options.name.singular !== model.name) continue;
for (const key of Object.keys(model.associations)) { for (const key of Object.keys(model.associations)) {
const association = model.associations[key]; const association = model.associations[key];
const { source, target } = association; const { source, target } = association;
@ -95,11 +94,15 @@ const register = (server, options = {}, next) => {
// build the methods for each model now that we've defined all the // build the methods for each model now that we've defined all the
// associations // associations
Object.keys(models).forEach((modelName) => { Object.keys(models).filter((modelName) => {
const model = models[modelName];
return model.options.name.singular === model.name;
}).forEach((modelName) => {
const model = models[modelName]; const model = models[modelName];
crud(server, model, options); crud(server, model, options);
}); });
next(); next();
}; };

View File

@ -3,21 +3,27 @@ import { notImplemented } from 'boom';
const sequelizeKeys = ['include', 'order', 'limit', 'offset']; const sequelizeKeys = ['include', 'order', 'limit', 'offset'];
const getModels = (request) => {
const noGetDb = typeof request.getDb !== 'function';
const noRequestModels = !request.models;
if (noGetDb && noRequestModels) {
return notImplemented('`request.getDb` or `request.models` are not defined.'
+ 'Be sure to load hapi-sequelize before hapi-sequelize-crud.');
}
const { models } = noGetDb ? request : request.getDb();
return models;
};
export const parseInclude = request => { export const parseInclude = request => {
const include = Array.isArray(request.query.include) const include = Array.isArray(request.query.include)
? request.query.include ? request.query.include
: [request.query.include] : [request.query.include]
; ;
const noGetDb = typeof request.getDb !== 'function'; const models = getModels(request);
const noRequestModels = !request.models; if (models.isBoom) return models;
if (noGetDb && noRequestModels) {
return notImplemented('`request.getDb` or `request.models` are not defined.'
+ 'Be sure to load hapi-sequelize before hapi-sequelize-crud.');
}
const { models } = noGetDb ? request : request.getDb();
return include.map(a => { return include.map(a => {
const singluarOrPluralMatch = Object.keys(models).find((modelName) => { const singluarOrPluralMatch = Object.keys(models).find((modelName) => {
@ -63,24 +69,40 @@ export const parseLimitAndOffset = (request) => {
return out; return out;
}; };
const parseOrderArray = (order, models) => {
return order.map((requestColumn) => {
if (Array.isArray(requestColumn)) {
return parseOrderArray(requestColumn, models);
}
let column;
try {
column = JSON.parse(requestColumn);
} catch (e) {
column = requestColumn;
}
if (column.model) column.model = models[column.model];
return column;
});
};
export const parseOrder = (request) => { export const parseOrder = (request) => {
const { order } = request.query; const { order } = request.query;
if (!order) return null; if (!order) return null;
const models = getModels(request);
if (models.isBoom) return models;
// transform to an array so sequelize will escape the input for us and // transform to an array so sequelize will escape the input for us and
// maintain security. See http://docs.sequelizejs.com/en/latest/docs/querying/#ordering // maintain security. See http://docs.sequelizejs.com/en/latest/docs/querying/#ordering
if (isString(order)) return [order.split(' ')]; const requestOrderColumns = isString(order) ? [order.split(' ')] : order;
for (const key of Object.keys(order)) { const parsedOrder = parseOrderArray(requestOrderColumns, models);
try {
order[key] = JSON.parse(order[key]);
} catch (e) {
//
}
}
return [order]; return parsedOrder;
}; };
export const getMethod = (model, association, plural = true, method = 'get') => { export const getMethod = (model, association, plural = true, method = 'get') => {

View File

@ -2,7 +2,8 @@ import test from 'ava';
import { parseLimitAndOffset, parseOrder, parseWhere } from './utils.js'; import { parseLimitAndOffset, parseOrder, parseWhere } from './utils.js';
test.beforeEach((t) => { test.beforeEach((t) => {
t.context.request = { query: {} }; const models = t.context.models = { User: {} };
t.context.request = { query: {}, models };
}); });
test('parseLimitAndOffset is a function', (t) => { test('parseLimitAndOffset is a function', (t) => {
@ -62,14 +63,13 @@ test('parseOrder returns order when a string', (t) => {
}); });
test('parseOrder returns order when json', (t) => { test('parseOrder returns order when json', (t) => {
const { request } = t.context; const { request,models } = t.context;
const order = [{ model: 'User' }, 'DESC'];
request.query.order = [JSON.stringify({ model: 'User' }), 'DESC']; request.query.order = [JSON.stringify({ model: 'User' }), 'DESC'];
request.query.thing = 'hi'; request.query.thing = 'hi';
t.deepEqual( t.deepEqual(
parseOrder(request) parseOrder(request)
, [order] , [{ model: models.User }, 'DESC']
); );
}); });

View File

@ -27,6 +27,13 @@ export default (sequelize, DataTypes) => {
name: 'notaname', name: 'notaname',
}, },
}, },
returnsAll: {
where: {
name: {
$ne: 'notaname',
},
},
},
}, },
}); });
}; };

View File

@ -58,11 +58,13 @@ export default (test) => {
const { Player, Team, City } = t.context.sequelize.models; const { Player, Team, City } = t.context.sequelize.models;
const city1 = await City.create({ name: 'Healdsburg' }); const city1 = await City.create({ name: 'Healdsburg' });
const team1 = await Team.create({ name: 'Baseballs', cityId: city1.id }); const team1 = await Team.create({ name: 'Baseballs', cityId: city1.id });
const team2 = await Team.create({ name: 'Footballs', cityId: city1.id });
const player1 = await Player.create({ const player1 = await Player.create({
name: 'Pinot', teamId: team1.id, active: true, name: 'Cat', teamId: team1.id, active: true,
}); });
const player2 = await Player.create({ name: 'Syrah', teamId: team1.id }); const player2 = await Player.create({ name: 'Pinot', teamId: team1.id });
t.context.instances = { city1, team1, player1, player2 }; const player3 = await Player.create({ name: 'Syrah', teamId: team2.id });
t.context.instances = { city1, team1, team2, player1, player2, player3 };
}); });
// kill the server so that we can exit and don't leak memory // kill the server so that we can exit and don't leak memory