diff --git a/README.md b/README.md index 315556acb..5f0851174 100644 --- a/README.md +++ b/README.md @@ -16,6 +16,12 @@ Dad jokes are all the rage these days! In this challenge, you will build a real wise-guy application. + + + + + + Users must be able to call the `[POST] /api/auth/register` endpoint to create a new account, and the `[POST] /api/auth/login` endpoint to get a token. We also need to make sure nobody without the token can call `[GET] /api/jokes` and gain access to our dad jokes. @@ -35,6 +41,15 @@ Your finished project must include all of the following requirements (further in - Codegrade is running some tests you cannot see in this repo. Make sure to comply with project instructions to the letter! - Do not exceed 2^8 rounds of hashing with `bcryptjs`. - If you use environment variables make sure to provide fallbacks in the code (e.g. `process.env.SECRET || "shh"`). + + + + + + + + + - You are welcome to create additional files but **do not move or rename existing files** or folders. - Do not alter your `package.json` file except to install extra libraries. Do not update existing packages. - The database already has the `users` table, but if you run into issues, the migration is available. diff --git a/api/auth/auth-router.js b/api/auth/auth-router.js index 47d8e51ae..1b27816de 100644 --- a/api/auth/auth-router.js +++ b/api/auth/auth-router.js @@ -1,7 +1,12 @@ const router = require('express').Router(); +const bcrypt = require('bcryptjs') +const {checkUsP, checkUsername, insert, checkLogin} = require('../middleware/users-middleware') +const jwt = require('jsonwebtoken') +const {JWT_SECRET} = require('../../api/secrets/index') -router.post('/register', (req, res) => { - res.end('implement register, please!'); +router.post('/register', checkUsP, checkUsername, async (req, res, next) => { + await insert(req, res, next) +}); /* IMPLEMENT You are welcome to build additional middlewares to help with the endpoint's functionality. @@ -27,10 +32,35 @@ router.post('/register', (req, res) => { 4- On FAILED registration due to the `username` being taken, the response body should include a string exactly as follows: "username taken". */ -}); +function buildToken (user) { + const payload = { + subject: user.id, + role: 'all access', + username: user.username + } + const options = { + expiresIn: '1d' + } + return jwt.sign(payload, JWT_SECRET, options) +} -router.post('/login', (req, res) => { - res.end('implement login, please!'); +router.post('/login', checkUsP, checkLogin, async (req, res, next) => { + try { + if (bcrypt.compareSync(req.body.password, req.user.password)) { + const token = buildToken(req.body) + res.status(200).json({ + message: `welcome, ${req.user.username}`, + token + }) + } + else { + next({status: 401, message: 'invalid credentials'}) + } + } + catch (error) { + next(error) + } +}); /* IMPLEMENT You are welcome to build additional middlewares to help with the endpoint's functionality. @@ -54,6 +84,6 @@ router.post('/login', (req, res) => { 4- On FAILED login due to `username` not existing in the db, or `password` being incorrect, the response body should include a string exactly as follows: "invalid credentials". */ -}); + module.exports = router; diff --git a/api/jokes/jokes-router.js b/api/jokes/jokes-router.js index f663f983c..97c62d6fd 100644 --- a/api/jokes/jokes-router.js +++ b/api/jokes/jokes-router.js @@ -1,8 +1,9 @@ // do not make changes to this file const router = require('express').Router(); const jokes = require('./jokes-data'); +const restricted = require('../middleware/restricted') -router.get('/', (req, res) => { +router.get('/', restricted, (req, res) => { res.status(200).json(jokes); }); diff --git a/api/middleware/restricted.js b/api/middleware/restricted.js index a690e961e..b39eca417 100644 --- a/api/middleware/restricted.js +++ b/api/middleware/restricted.js @@ -1,5 +1,26 @@ -module.exports = (req, res, next) => { - next(); +const {secret} = require('../secrets/index') +const jwt = require('jsonwebtoken') + + +module.exports = async (req, res, next) => { + const token = req.headers.authorization + return res.status(406).json({message: token}) + // if (token) { + // await jwt.verify(token, secret, (error, decoded) => { + // if (error) { + // next({status: 401, message: 'token invalid', er: error}) + // } + // else { + // req.decodedJwt = decoded + // next() + // } + // }) + // } + // else { + // next({status:401, message: 'token required'}) + // } + + /* IMPLEMENT diff --git a/api/middleware/users-middleware.js b/api/middleware/users-middleware.js new file mode 100644 index 000000000..6a5d5fd4a --- /dev/null +++ b/api/middleware/users-middleware.js @@ -0,0 +1,56 @@ +const db = require('../../data/dbConfig') +const bcrypt = require('bcryptjs') + + +async function checkUsername (req, res, next) { + const username = req.body.username + const exists = await db('users').select('*').where('username', username).first() + if (exists) { + next({status: 400, message: 'username taken'}) + } + else next() +} + +async function checkLogin (req, res, next) { + try { + const user = await db('users').select('*').where('username', req.body.username).first() + if (user) { + req.user = user + next() + } + else { + next({status: 401, message: 'invalid credentials'}) + } + } + catch (error) { + next(error) + } +} + +async function checkUsP (req, res, next) { + const {username, password} = req.body + + if (!username || !password) { + next({status: 402, message: 'username and password required'}) + } + else next() +} + +async function insert (req, res, next) { + try { + let {username, password} = req.body + password = await bcrypt.hashSync(password, 8) + + await db('users').insert({username, password}) + + const user = await db('users').select('*').where('username', username).first() + res.status(201).json(user) + } + catch (error) { + next(error) + } +} + +module.exports = { + checkUsername, insert, checkUsP, checkLogin +} \ No newline at end of file diff --git a/api/secrets/index.js b/api/secrets/index.js new file mode 100644 index 000000000..9f0158f59 --- /dev/null +++ b/api/secrets/index.js @@ -0,0 +1,5 @@ +const JWT_SECRET = process.env.JWT_SECRET || 'shh' + +module.exports = { + JWT_SECRET +} \ No newline at end of file diff --git a/api/server.js b/api/server.js index 33320b871..7eb8f4c97 100644 --- a/api/server.js +++ b/api/server.js @@ -16,4 +16,16 @@ server.use(express.json()); server.use('/api/auth', authRouter); server.use('/api/jokes', restrict, jokesRouter); // only logged-in users should have access! -module.exports = server; + +server.use('*', (req, res, next) => { + next({status: 404, message: 'not found'}) +}) + +server.use( (error, req, res, next) => { //eslint-disable-line + res.status(error.status || 500).json({ + message: error.message || 'error', + stack: error.stack + }) +}) + +module.exports = server; \ No newline at end of file diff --git a/api/server.test.js b/api/server.test.js index 96965c559..12451d455 100644 --- a/api/server.test.js +++ b/api/server.test.js @@ -1,4 +1,90 @@ -// Write your tests here + +const db = require('../data/dbConfig') +const request = require('supertest') +const server = require('../api/server') + +beforeAll( async () => { + await db.migrate.rollback() + await db.migrate.latest() +}) + +beforeEach( async () => { + await db.seed.run() +}) + +// afterAll( async () => { +// await db.destroy() +// }) + + test('sanity', () => { - expect(true).toBe(false) + expect(true).toBe(true) +}) + + + + + + +describe('[POST] /api/auth/register', () => { + test('gets success status', async () => { + const creds = {username: 'hello', password: '1234'} + const res = await request(server).post('/api/auth/register').send(creds) + expect(res.status).toBe(201) + }) + test('adds credentials to database', async () => { + const creds = {username: 'hello', password: '1234'} + await request(server).post('/api/auth/register').send(creds) + const res = await db('users').select('username').where('username', creds.username).first() + expect(res.username).toBe(creds.username) + }) +}) + + + + + + +describe('[POST] /api/auth/login', () => { + test('gets a success status', async () => { + const creds = {username: 'hello', password: '1234'} + const res1 = await request(server).post('/api/auth/register').send(creds) + expect(res1.status).toBe(201) + const res = await request(server).post('/api/auth/login').send(creds) + expect(res.status).toBe(200) + }) + test('valid login gets a token', async () => { + const creds = {username: 'hello', password: '1234'} + await request(server).post('/api/auth/register').send(creds) + const res = await request(server).post('/api/auth/login').send(creds) + expect(res.body.token).toBeDefined() + }) +}) + + + + + + +describe('[GET] /api/jokes', () => { + test.only('valid token gets the jokes', async () => { + + const creds = {username: 'hello', password: '1234'} + const res1 = await request(server).post('/api/auth/register').send(creds) + expect(res1.status).toBe(201) + + const res2 = await request(server).post('/api/auth/login').send(creds) + expect(res2.body.token).toBeDefined() + + const res3 = await request(server).get('/api/jokes').set('Authorization', res2.body.token) + expect(res3.body.message).toHaveLength(3) + }) + + + + test('no token gets error', async () => { + const res = await request(server).get('/api/jokes') + expect(res.body.jokes).not.toBeDefined() + expect(res.status).toBe(401) + }) }) diff --git a/data/migrations/20201123181212_users.js b/data/migrations/20201123181212_users.js index 98f96ee52..960e9fd02 100644 --- a/data/migrations/20201123181212_users.js +++ b/data/migrations/20201123181212_users.js @@ -1,11 +1,19 @@ -exports.up = function (knex) { - return knex.schema.createTable('users', users => { - users.increments(); - users.string('username', 255).notNullable().unique(); - users.string('password', 255).notNullable(); - }); +exports.up = async function (knex) { + + await knex.schema.createTable('users', users => { + users.increments(); + users.string('username', 255).notNullable().unique(); + users.string('password', 255).notNullable(); + }); + + await knex.schema.createTable('jokes', table => { + table.increments('db_id') + table.string('id').notNullable() + table.string('joke', 400).notNullable().unique() + }) }; -exports.down = function (knex) { - return knex.schema.dropTableIfExists('users'); +exports.down = async function (knex) { + await knex.schema.dropTableIfExists('users'); + await knex.schema.dropTableIfExists('jokes'); }; diff --git a/data/migrations/20231121002935_jokes-migrate.js b/data/migrations/20231121002935_jokes-migrate.js new file mode 100644 index 000000000..143f816c4 --- /dev/null +++ b/data/migrations/20231121002935_jokes-migrate.js @@ -0,0 +1,15 @@ +/** + * @param { import("knex").Knex } knex + * @returns { Promise } + */ +exports.up = function(knex) { + +}; + +/** + * @param { import("knex").Knex } knex + * @returns { Promise } + */ +exports.down = function(knex) { + +}; diff --git a/data/seeds/initial-seed.js b/data/seeds/initial-seed.js new file mode 100644 index 000000000..f72680ba5 --- /dev/null +++ b/data/seeds/initial-seed.js @@ -0,0 +1,8 @@ +const jokes = require('../../api/jokes/jokes-data') + +exports.seed = async function(knex) { + // Deletes ALL existing entries + await knex('users').truncate() + await knex('jokes').truncate() + await knex('jokes').insert(jokes); +}; diff --git a/data/seeds/jokes-seed.js b/data/seeds/jokes-seed.js new file mode 100644 index 000000000..d0e654031 --- /dev/null +++ b/data/seeds/jokes-seed.js @@ -0,0 +1,13 @@ +/** + * @param { import("knex").Knex } knex + * @returns { Promise } + */ +exports.seed = async function(knex) { + // Deletes ALL existing entries + await knex('table_name').del() + await knex('table_name').insert([ + {id: 1, colName: 'rowValue1'}, + {id: 2, colName: 'rowValue2'}, + {id: 3, colName: 'rowValue3'} + ]); +}; diff --git a/knexfile.js b/knexfile.js index 34b52cfea..f68ccdc4b 100644 --- a/knexfile.js +++ b/knexfile.js @@ -3,6 +3,7 @@ const sharedConfig = { client: 'sqlite3', useNullAsDefault: true, migrations: { directory: './data/migrations' }, + seeds: { directory: './data/seeds' }, pool: { afterCreate: (conn, done) => conn.run('PRAGMA foreign_keys = ON', done) }, } @@ -10,7 +11,7 @@ module.exports = { development: { ...sharedConfig, connection: { filename: './data/auth.db3' }, - seeds: { directory: './data/seeds' }, + }, testing: { ...sharedConfig, diff --git a/package-lock.json b/package-lock.json index 7e4320161..8b7d47fdf 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9,9 +9,11 @@ "version": "0.0.1", "license": "ISC", "dependencies": { + "bcryptjs": "^2.4.3", "cors": "2.8.5", "express": "4.18.1", "helmet": "5.0.2", + "jsonwebtoken": "^9.0.2", "knex": "2.0.0", "sqlite3": "5.0.8" }, @@ -1586,6 +1588,11 @@ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" }, + "node_modules/bcryptjs": { + "version": "2.4.3", + "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz", + "integrity": "sha512-V/Hy/X9Vt7f3BbPJEi8BdVFMByHi+jNXrYkW3huaybV/kQ0KJg0Y6PkEMbn+zeT+i+SiKZ/HMqJGIIt4LZDqNQ==" + }, "node_modules/binary-extensions": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", @@ -1724,6 +1731,11 @@ "node-int64": "^0.4.0" } }, + "node_modules/buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==" + }, "node_modules/buffer-from": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", @@ -2359,6 +2371,14 @@ "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", "dev": true }, + "node_modules/ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "dependencies": { + "safe-buffer": "^5.0.1" + } + }, "node_modules/ee-first": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", @@ -4712,6 +4732,60 @@ "node": ">=6" } }, + "node_modules/jsonwebtoken": { + "version": "9.0.2", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz", + "integrity": "sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==", + "dependencies": { + "jws": "^3.2.2", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=12", + "npm": ">=6" + } + }, + "node_modules/jsonwebtoken/node_modules/semver": { + "version": "7.5.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", + "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/jwa": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", + "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", + "dependencies": { + "buffer-equal-constant-time": "1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/jws": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", + "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", + "dependencies": { + "jwa": "^1.4.1", + "safe-buffer": "^5.0.1" + } + }, "node_modules/keyv": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", @@ -4845,12 +4919,47 @@ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" }, + "node_modules/lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==" + }, + "node_modules/lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==" + }, + "node_modules/lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==" + }, + "node_modules/lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==" + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==" + }, + "node_modules/lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==" + }, "node_modules/lodash.merge": { "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", "dev": true }, + "node_modules/lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==" + }, "node_modules/lowercase-keys": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", @@ -8226,6 +8335,11 @@ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" }, + "bcryptjs": { + "version": "2.4.3", + "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz", + "integrity": "sha512-V/Hy/X9Vt7f3BbPJEi8BdVFMByHi+jNXrYkW3huaybV/kQ0KJg0Y6PkEMbn+zeT+i+SiKZ/HMqJGIIt4LZDqNQ==" + }, "binary-extensions": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", @@ -8330,6 +8444,11 @@ "node-int64": "^0.4.0" } }, + "buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==" + }, "buffer-from": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", @@ -8813,6 +8932,14 @@ "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", "dev": true }, + "ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "requires": { + "safe-buffer": "^5.0.1" + } + }, "ee-first": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", @@ -10620,6 +10747,52 @@ "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", "dev": true }, + "jsonwebtoken": { + "version": "9.0.2", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz", + "integrity": "sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==", + "requires": { + "jws": "^3.2.2", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^7.5.4" + }, + "dependencies": { + "semver": { + "version": "7.5.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", + "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", + "requires": { + "lru-cache": "^6.0.0" + } + } + } + }, + "jwa": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", + "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", + "requires": { + "buffer-equal-constant-time": "1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "^5.0.1" + } + }, + "jws": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", + "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", + "requires": { + "jwa": "^1.4.1", + "safe-buffer": "^5.0.1" + } + }, "keyv": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", @@ -10708,12 +10881,47 @@ "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" }, + "lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==" + }, + "lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==" + }, + "lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==" + }, + "lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==" + }, + "lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==" + }, + "lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==" + }, "lodash.merge": { "version": "4.6.2", "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", "dev": true }, + "lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==" + }, "lowercase-keys": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", diff --git a/package.json b/package.json index 9506c6c20..2391c4810 100644 --- a/package.json +++ b/package.json @@ -6,7 +6,10 @@ "start": "node index.js", "server": "nodemon index.js", "migrate": "knex migrate:latest", - "test": "cross-env NODE_ENV=testing jest --verbose --runInBand --silent" + "rollback": "knex migrate:rollback", + "seed": "knex seed:run", + "test": "cross-env NODE_ENV=testing jest --verbose --runInBand --silent --watchAll", + "resetdb": "npm run rollback && npm run migrate && npm run seed" }, "repository": { "type": "git", @@ -14,9 +17,11 @@ }, "license": "ISC", "dependencies": { + "bcryptjs": "^2.4.3", "cors": "2.8.5", "express": "4.18.1", "helmet": "5.0.2", + "jsonwebtoken": "^9.0.2", "knex": "2.0.0", "sqlite3": "5.0.8" }, @@ -28,4 +33,4 @@ "nodemon": "2.0.16", "supertest": "6.2.3" } -} +} \ No newline at end of file