diff --git a/config.json.example b/config.json.example index 5b2a49fdf244d7a7b54ce749d8eabe7546e5d3e1..ea9bd2f3a7c826ed36999cb360b748801d1445ef 100644 --- a/config.json.example +++ b/config.json.example @@ -1,21 +1,71 @@ { - "port": 3000, - "ip": "127.0.0.1", - "debug" : false, - "monetdb": { - "host": "simcaqdb3.c3sl.ufpr.br", - "port": 50000, - "dbname": "simcaq_dev", - "user": "monetdb", - "password":"monetdb", - "nrConnections": "16" + "development": + { + "port": 3000, + "ip": "127.0.0.1", + "debug" : true, + "monetdb": { + "host": "simcaqdb3.c3sl.ufpr.br", + "port": 50000, + "dbname": "simcaq_dev", + "user": "monetdb", + "password":"monetdb", + "nrConnections": "4" + }, + "mongodb" : { + "uri": "mongodb://localhost/users", + "secret": "SimCAQC3SL" + }, + "default": { + "api": { + "version" : "v1" + } + } }, - "mongodb" : { - "uri": "mongodb://localhost/users" + "test": + { + "port": 3000, + "ip": "127.0.0.1", + "debug" : true, + "monetdb": { + "host": "simcaqdb3.c3sl.ufpr.br", + "port": 50000, + "dbname": "simcaq_dev", + "user": "monetdb", + "password":"monetdb", + "nrConnections": "4" + }, + "mongodb" : { + "uri": "mongodb://localhost/test_users", + "secret": "SimCAQC3SL" + }, + "default": { + "api": { + "version" : "v1" + } + } }, - "default": { - "api": { - "version" : "v1" + "production": + { + "port": 3000, + "ip": "127.0.0.1", + "debug" : false, + "monetdb": { + "host": "simcaqdb3.c3sl.ufpr.br", + "port": 50000, + "dbname": "simcaq_dev", + "user": "monetdb", + "password":"monetdb", + "nrConnections": "4" + }, + "mongodb" : { + "uri": "mongodb://localhost/users", + "secret": "SimCAQC3SL" + }, + "default": { + "api": { + "version" : "v1" + } } } -} \ No newline at end of file +} diff --git a/gulpfile.babel.js b/gulpfile.babel.js index 563dfac93e840a304a15e10769deb69db755195d..acc0e918ad5bea19aad613165a84e547648520a6 100644 --- a/gulpfile.babel.js +++ b/gulpfile.babel.js @@ -73,7 +73,7 @@ gulp.task('pre-test', () => { gulp.task('test', ['pre-test'], () => { process.chdir('build'); - gulp.src('test/**/*.js', {read: false}) + gulp.src(['test/**/*.js'], {read: false}) .pipe(mocha({timeout: 15000})) .pipe(istanbul.writeReports()) .pipe(istanbul.enforceThresholds({ diff --git a/package.json b/package.json index acc29cd2e3d6ff9a605fde13cfaa6774243b6307..a7056dc915f19875538b5f4de96bda8d4fa764d9 100644 --- a/package.json +++ b/package.json @@ -10,6 +10,7 @@ }, "dependencies": { "apicache": "0.0.14", + "bcrypt": "^0.8.7", "body-parser": "^1.13.1", "chai": "^3.5.0", "chai-http": "^3.0.0", @@ -23,11 +24,13 @@ "faker": "^2.1.5", "forever": "^0.15.2", "js2xmlparser": "^1.0.0", + "jwt-simple": "^0.5.0", "method-override": "^2.3.3", "mocha": "^2.5.3", "monetdb-pool": "0.0.8", "mongoose": "^4.6.0", "nconf": "^0.6.x", + "passport": "^0.3.2", "squel": "^5.4.2", "winston": "^2.2.0" }, diff --git a/src/libs/app.js b/src/libs/app.js index 37e02f87922011bf2c05f39b638b57a5c3c52e0c..18b46cf3f195879ce25d1edf2bcaa39a08afb6f3 100644 --- a/src/libs/app.js +++ b/src/libs/app.js @@ -10,12 +10,15 @@ const libs = `${process.cwd()}/libs`; const log = require(`${libs}/log`)(module); +process.env.NODE_ENV = process.env.NODE_ENV || 'development'; const config = require(`${libs}/config`); -const cache = require('apicache').options({ debug: config.get('debug') }).middleware; +const cache = require('apicache').options({ debug: config.debug }).middleware; const app = express(); -const api = require(`${libs}/routes/api`); +const api = require('./routes/api'); + +const passport = require('passport'); const mongoose = require(`${libs}/db/mongoose`); @@ -27,8 +30,6 @@ app.use(cookieParser()); // Enable Cross-Origin Resource Sharing (CORS) app.use(cors()); app.use(methodOverride()); -// Enable cache for 1 day -app.use(cache('1 day')); // Enable maximum compression app.use(compression(9)); // Middleware tha adds the squel object to req diff --git a/src/libs/config.js b/src/libs/config.js index 5aea5a9ca50efd0dac2253dd23690229431236dc..2c07d19e94ba2505205d613a9ac6d5596c388502 100644 --- a/src/libs/config.js +++ b/src/libs/config.js @@ -1,8 +1,5 @@ -const nconf = require('nconf'); +let conf = require(`${process.cwd()}/config.json`); -// Exports the config.json as an object with get functions -nconf.argv() - .env() - .file({ file: `${process.cwd()}/config.json` }); +conf = conf[process.env.NODE_ENV]; -module.exports = nconf; +module.exports = conf; diff --git a/src/libs/db/monet.js b/src/libs/db/monet.js index dd7e620adcf5b6a1dbe29358697fa71a3970ea03..d19fb37dfac5ed68cc15c1762a822818dbbc82da 100644 --- a/src/libs/db/monet.js +++ b/src/libs/db/monet.js @@ -6,16 +6,16 @@ const config = require(`${libs}/config`); // Connection options const poolOptions = { - nrConnections: config.get('monetdb:nrConnections'), + nrConnections: config.monetdb.nrConnections, }; // Configuration options const options = { - host: config.get('monetdb:host'), - port: config.get('monetdb:port'), - dbname: config.get('monetdb:dbname'), - user: config.get('monetdb:user'), - password: config.get('monetdb:password'), + host: config.monetdb.host, + port: config.monetdb.port, + dbname: config.monetdb.dbname, + user: config.monetdb.user, + password: config.monetdb.password, }; // Connection singleton diff --git a/src/libs/db/mongoose.js b/src/libs/db/mongoose.js index 30b4d4930c203537335c46b14bfb5e2958ace6bb..d8594158aba29c2c732e8e348bca7930fbb84e73 100644 --- a/src/libs/db/mongoose.js +++ b/src/libs/db/mongoose.js @@ -6,9 +6,12 @@ const log = require(`${libs}/log`)(module); const mongoose = require('mongoose'); +mongoose.Promise = global.Promise; + module.exports = () => { - const mongoUri = process.env.MONGO_URI || config.get('mongodb:uri'); - log.debug(`Connecting to MongDB on URI ${mongoUri}`); + // Get mongodb URI (ip and port) in config file + const mongoUri = process.env.MONGO_URI || config.mongodb.uri; + log.debug(`Connecting to MongoDB on URI ${mongoUri}`); // Connection singleton const db = mongoose.connect(mongoUri); diff --git a/src/libs/log.js b/src/libs/log.js index fbe7b1394464e06611cbcd91d69e33c022846923..ddf0303f8481514b047aa6678894316500d25a96 100644 --- a/src/libs/log.js +++ b/src/libs/log.js @@ -47,7 +47,7 @@ function logger(module) { ], exitOnError: false, }); - if (!config.get('debug')) { + if (!config.debug) { log.remove('debug-log'); } return log; diff --git a/src/libs/middlewares/passport.js b/src/libs/middlewares/passport.js new file mode 100644 index 0000000000000000000000000000000000000000..c939926099b845b641268d66e9d2bccf9085a043 --- /dev/null +++ b/src/libs/middlewares/passport.js @@ -0,0 +1,54 @@ +const JwtStrategy = require('passport-jwt').Strategy; +const ExtractJwt = require('passport-jwt').ExtractJwt; +const libs = `${process.cwd()}/libs`; +const config = require(`${libs}/config`); +const User = require(`${libs}/models/user`) + +module.exports = function(passport){ + var opts = {}; + opts.jwtFromRequest = ExtractJwt.fromAuthHeader(); + opts.secretOrKey = config.get('mongodb:secret'); + passport.use(new JwtStrategy(opts, function(jwt_payload, done){ + User.find({email: jwt_payload.email}, function(err, user){ + if (err) { + return done(err); + } + + if (!user) { + return done(null, false, {message: 'Unknown user'}); + } + + return done(null, user); + }); + })); +}; + +/* To check if a user has access to a route, one must use passport.authenticate() specifying 'JWT' as the strategy in the route declaration, like so: +//pass passportfor configuration +require('./config/passport')(passport); + +app.post('/route', passport.authenticate('jwt', { session: false}), function(req, res) { }); + +the user object is then accessible via req.user +---- + +Another way to check if a user is authenticated, is to check the request header for the json web token, like so: + +getToken = function (headers) { + if (headers && headers.authorization) { + var parted = headers.authorization.split(' '); + if (parted.length === 2) { + return parted[1]; + } else { + return null; + } + } else { + return null; + } +}; + +var token = getToken(req.headers); + if (token) { + var decoded = jwt.decode(token, config.get(mongodb.secret)); + } + */ diff --git a/src/libs/models/simulation.js b/src/libs/models/simulation.js new file mode 100644 index 0000000000000000000000000000000000000000..e70c93a9f88dea76d017744b15f3b3fe61b5ca21 --- /dev/null +++ b/src/libs/models/simulation.js @@ -0,0 +1,115 @@ +const mongoose = require('mongoose') + +const libs = `${process.cwd()}/libs`; +const log = require(`${libs}/log`)(module); + +const Schema = mongoose.Schema; + +// Should define this somewhere else +const MAX_SIMULATION_TIME = 10; + + +let SimulationSchema = new Schema({ + name: { + type: String, + required: true, + }, + location: Object, + time: Number, + failure_rate: Array, + goals_care: Array, + goals_inclusion: Array, + enrollments: Array, + timestamp: Date, +}); + +SimulationSchema.methods.setTime = function (t) { + t = parseInt(t, 10); + if (!t || t > MAX_SIMULATION_TIME) { + // Throw an error? + return false; + } + this.time = t; + return true; +}; +SimulationSchema.methods.setLocation = function (l) { + // Should sanitize + this.location = l; + return true; +}; +SimulationSchema.methods.setFailureRate = function (fr) { + // Should sanitize + this.failure_rate = fr; + return true; +}; +SimulationSchema.methods.setCareGoals = function (g) { + // Should sanitize + this.goals_care = g; + return true; +}; +SimulationSchema.methods.setInclusionGoals = function (g) { + // Should sanitize + this.goals_inclusion = g; + return true; +}; +SimulationSchema.methods.setEnrollments = function (e) { + try{ + e = JSON.parse(e); + } catch (err) { + return false; + } + let success = true; + for(let i=0; i<e.length; i++){ + if(!(e[i] instanceof Array)){ + return false; + } + if(e[i].length !== this.time){ + return false; + } + e[i].forEach((n, i, array) => { + if(n !== parseInt(n, 10)){ + success = false; + } + }); + + } + if (success) this.enrollments = e; + + return success; +} +SimulationSchema.methods.update = function (property, value) { + let success = true; + switch(property){ + case 'time': + if (!this.setTime(value)) success = false; + break; + case 'location': + if (!this.setLocation(value)) success = false; + break; + case 'failure_rate': + if (!this.setFailureRate(value)) success = false; + break; + case 'goals_care': + if (!this.setCareGoals(value)) success = false; + break; + case 'goals_inclusion': + if (!this.setInclusionGoals(value)) success = false; + break; + case 'enrollments': + if (!this.setEnrollments(value)) success = false; + break; + case 'name': + this.name = value; + break; + } + return success; +}; + +SimulationSchema.methods.run = function () { + /* Runs the Simulation with given parameters */ + // if (!this.name || !this.location || !this.time) { + // console.log('This is supposed to be an error.'); + // } +}; + +module.exports = mongoose.model('Simulation', SimulationSchema); diff --git a/src/libs/models/user.js b/src/libs/models/user.js new file mode 100644 index 0000000000000000000000000000000000000000..8c4f92a204ef9c3b410fb656c16c4af9ad6c7152 --- /dev/null +++ b/src/libs/models/user.js @@ -0,0 +1,89 @@ +const mongoose = require('mongoose'); +const bcrypt = require('bcrypt'); +const libs = `${process.cwd()}/libs`; +const log = require(`${libs}/log`)(module); +const Schema = mongoose.Schema; + +// set up a mongoose model +var UserSchema = new Schema({ + email: { + type: String, + unique: true, + required: true + }, + password: { + type: String, + required: true + }, + name: { + type: String, + required: true + }, + cpf:{ + type: String, + unique: true, + required: true + }, + schooling: { + type: String, + required: true + }, + course: { + type: String, + }, + segment: { + type: String, + required: true + }, + role: { + type: String, + required: true + }, + institution_name: { + type: String, + required: true + }, + state: { + type: String, + required: true + }, + city: { + type: String, + required: true + }, + receive_emails: { + type: Boolean + } + +}); + +UserSchema.pre('save', function (next) { + var user = this; + if (this.isModified('password') || this.isNew) { + bcrypt.genSalt(10, function (err, salt) { + if (err) { + return next(err); + } + bcrypt.hash(user.password, salt, function (err, hash) { + if (err) { + return next(err); + } + user.password = hash; + next(); + }); + }); + } else { + return next(); + } +}); + +UserSchema.methods.comparePassword = function (passw, cb) { + bcrypt.compare(passw, this.password, function (err, isMatch) { + if (err) { + return cb(err); + } + cb(null, isMatch); + }); +}; + +module.exports = mongoose.model('User', UserSchema); diff --git a/src/libs/routes/api.js b/src/libs/routes/api.js index 0522a8d0a5cfda2bb3a6f663ed1a44d683e1ff54..2c29a24680987a9b6c572cf5fa4ba86bc0b0c83c 100644 --- a/src/libs/routes/api.js +++ b/src/libs/routes/api.js @@ -2,6 +2,12 @@ const express = require('express'); const api = express(); +const libs = `${process.cwd()}/libs`; + +const config = require(`${libs}/config`); + +const cache = require('apicache').options({ debug: config.debug }).middleware; + const enrollment = require('./enrollment'); const state = require('./state'); @@ -14,16 +20,22 @@ const school = require('./school'); const location = require('./location'); +const simulation = require('./simulation'); + +const user = require('./user'); + api.get('/', (req, res) => { res.json({ msg: 'SimCAQ API is running' }); }); // mount API routes -api.use('/enrollment', enrollment); -api.use('/state', state); -api.use('/region', region); -api.use('/city', city); -api.use('/school', school); -api.use('/location', location); +api.use('/user', user); +api.use('/simulation', simulation); +api.use('/enrollment', cache('1 day'), enrollment); +api.use('/state', cache('15 day'), state); +api.use('/region', cache('15 day'), region); +api.use('/city', cache('15 day'), city); +api.use('/school', cache('15 day'), school); +api.use('/location', cache('1 day'), location); module.exports = api; diff --git a/src/libs/routes/simulation.js b/src/libs/routes/simulation.js index d197b39d1c107935bef0775bcfce31ca4b7b75b5..6469d4d6b3df70077b9f38fca96c6ae479133022 100644 --- a/src/libs/routes/simulation.js +++ b/src/libs/routes/simulation.js @@ -1,47 +1,147 @@ const express = require('express'); +const simulationApp = express(); + const libs = `${process.cwd()}/libs`; +const log = require(`${libs}/log`)(module); + const squel = require('squel'); const query = require(`${libs}/middlewares/query`); const response = require(`${libs}/middlewares/response`); -const simulationApp = express(); - -simulationApp.get('/', (req, res, next) => { - req.sql = squel.select().from('simulacao').toParam(); - next(); -}, query, response('simulacao')); - -simulationApp.get('/:uid:/all', (req, res, next) => { - // TODO: implement user checking - - req.sql = squel.select().from('simulacao').toParam(); - next(); -}, query, response('simulacao')); - -simulationApp.get('/:uid:/:simulation_id:/view', (req, res, next) => { - // TODO: implement checking if the simulation belongs to the current user - - req.sql = squel.select().from('simulacao').toParam(); - next(); +const Simulation = require(`${libs}/models/simulation`); + +simulationApp.get('/', (req, res) => { + let out = { success: true, msg: 'controller working' }; + out.result = new Array() + Simulation.find({}, function(err, sims) { + sims.forEach((sim) => { + out.result.push(sim); + }); + res.send(out); + }); }); -simulationApp.get('/:uid:/:simulation_id:/delete', (req, res, next) => { - // TODO: implement checking if the simulation belongs to the current user +simulationApp.post('/', (req, res, next) => { + // This method must associate new entry with user. + /* Creates a new simulation. Requires a name. */ + if (!req.body.name) { + res.send({ success: false, msg: 'Must send a name for new entry' }); + } else { + next(); + } +}, (req, res) => { + let currentdate = new Date(); + let newSimulation = new Simulation({ + name: req.body.name, + timestamp: currentdate.getDate(), + }); + newSimulation.save((err) => { + if (err) { + res.send({ success: false, msg: err }); + } else { + res.send({ + success: true, + msg: 'new sim created', + id: newSimulation._id, + }); + } + }); +}); - req.sql = squel.select().from('simulacao').toParam(); - next(); +simulationApp.get('/:id', (req, res) => { + /* Can be used to check simulation construction status */ + Simulation.findById(req.params.id, (err, simulation) => { + log.debug(req.params.id); + if (err) { + res.send({ success: false, msg: err }); + return; + } + + if (!simulation) { + res.send({ success: false, msg: 'Entry not found' }); + } else { + res.send(simulation); + } + }); }); -simulationApp.get('/:uid:/:simulation_id:/edit', (req, res, next) => { - // TODO: implement checking if the simulation belongs to the current user +simulationApp.post('/:id', (req, res, next) => { + if (!Object.keys(req.body).length) { + res.send({ success: false, msg: 'No field informed to update' }); + } else { + next(); + } +}, (req, res, next) => { + let simulation = Simulation.findById(req.params.id, (err, simulation) => { + if (err) { + res.send({ success: false, msg: err }); + } else { + if (!simulation) { + res.send({ success: false, msg: 'Entry not found' }); + } else { + req.simulation = simulation; + next(); + } + } + }); +}, (req, res) => { + for (let property in req.body) { + if (Simulation.schema.tree.hasOwnProperty(property)) { + if(!req.simulation.update(property, req.body[property])){ + res.send({ + success: false, + msg: 'Invalid format for ' + property, + }); + return ; + } + } else { + res.send({ + success: false, + msg: 'Unknown property ' + property, + }); + return ; + } + } + req.simulation.save((err) => { + if (err) { + res.send({ success: false, msg: err }); + } else { + res.send({ + success: true, + msg: 'sim updated', + id: req.simulation._id, + }); + } + }); +}); - req.sql = squel.select().from('simulacao').toParam(); - next(); +simulationApp.delete('/:id', (req, res, next) => { + let simulation = Simulation.findById(req.params.id, (err, simulation) => { + if (err) { + res.send({ success: false, msg: err }); + } else { + if (!simulation) { + res.send({ success: false, msg: 'Entry not found' }); + } else { + next(); + } + } + }); +}, (req, res) => { + Simulation.remove({"_id": req.params.id}, (err) => { + if (err) { + res.send({ success: false, msg: err }); + } else { + res.send({ + success: true, + msg: 'sim removed', + }); + } + }); }); module.exports = simulationApp; - diff --git a/src/libs/routes/user.js b/src/libs/routes/user.js new file mode 100644 index 0000000000000000000000000000000000000000..eb1732b18f6a04a74c6799ac81d7a4ca027c8e82 --- /dev/null +++ b/src/libs/routes/user.js @@ -0,0 +1,194 @@ +const express = require('express'); + +const userApp = express(); + +const libs = `${process.cwd()}/libs`; + +const config = require(`${libs}/config`); + +const log = require(`${libs}/log`)(module); + +const User = require(`${libs}/models/user`); + +const jwt = require('jwt-simple'); + +const required_fields = ["email", "password", "name", "cpf", "schooling", "segment", "role", "institution_name", "state", "city"]; + + +function emailSyntax(email) { + const regex = /^(([^<>()\[\]\.,;:\s@\"]+(\.[^<>()\[\]\.,;:\s@\"]+)*)|(\".+\"))@(([^<>()[\]\.,;:\s@\"]+\.)+[^<>()[\]\.,;:\s@\"]{2,})$/i; + return regex.test(email); +} + +userApp.post('/', (req, res, next) => { + if (!req.body.email) { + res.json({success: false, msg: 'O campo Email é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.password) { + res.json({success: false, msg: 'O campo Senha é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if(!emailSyntax(req.body.email)){ + res.json({success: false, msg: 'O email informado é inválido.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.name) { + res.json({success: false, msg: 'O campo Nome é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.cpf) { + res.json({success: false, msg: 'O campo CPF é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.schooling) { + res.json({success: false, msg: 'O campo Escolaridade é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.segment) { + res.json({success: false, msg: 'O campo Segmento é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.role) { + res.json({success: false, msg: 'O campo Função é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.institution_name) { + res.json({success: false, msg: 'O campo Instituição em que trabalha é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.city) { + res.json({success: false, msg: 'O campo Cidade é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.state) { + res.json({success: false, msg: 'O campo Estado é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + User.count({'email': req.body.email}, function(err, count){ + if (err){ + log.error('MongoDB error: ' + err); + res.json({success: false, msg: 'Um erro ocorreu no banco de dados.'}); + } + if(count){ + res.json({success: false, msg: 'O email informado já está cadastrado.'}); + } else { + next(); + } + }); + +}, (req, res, next) => { + User.count({'cpf': req.body.cpf}, function(err, count){ + if (err){ + log.error('MongoDB error: ' + err); + res.json({success: false, msg: 'Um erro ocorreu no banco de dados.'}); + } + if(count){ + res.json({success: false, msg: 'O CPF informado já está cadastrado.'}); + } else { + next(); + } + }); + +}, (req, res, next) => { + var newUser = new User({ + email: req.body.email, + password: req.body.password, + name: req.body.name, + cpf: req.body.cpf, + schooling: req.body.schooling, + course: req.body.course, + segment: req.body.segment, + role: req.body.role, + institution_name: req.body.institution_name, + state: req.body.state, + city: req.body.city, + receive_emails: req.body.receive_emails + }); + + // save the user + newUser.save((err) => { + if (err) { + res.json({success: false, msg: 'Um erro ocorreu no banco de dados.'}); + } else { + res.json({success: true, msg: 'Usuário cadastrado com sucesso!'}); + } + }); +}); + +userApp.post('/authenticate', (req, res, next) => { + if (!req.body.email) { + res.json({success: false, msg: 'O campo Email é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + if (!req.body.password) { + res.json({success: false, msg: 'O campo Senha é obrigatório.'}); + } else { + next(); + } + +}, (req, res, next) => { + User.findOne({ + email: req.body.email + }, (err, user) => { + if (err) throw err; + + if(!user){ + res.json({success: false, msg: 'O Email informado não está cadastrado.'}); + } + else { + user.comparePassword(req.body.password, (err, isMatch) => { + if (isMatch && !err) { + var secret = config.mongodb.secret; + + // if user is found and password is right create a token + var token = jwt.encode(user, secret); + + //returns user info including token as json + res.json({success: true, token: 'JWT ' + token, msg: 'Usuário autenticado com sucesso'}); + } + else { + res.json({success: false, msg: 'A Senha informada é inválida.'}); + } + }); + } + }); +}); + +module.exports = userApp; diff --git a/src/server.js b/src/server.js index 0eef8dfa5fb4764522f0b1c0be02da239603d347..4e097f28a4053050757c6329e19e225e0424b961 100644 --- a/src/server.js +++ b/src/server.js @@ -8,12 +8,14 @@ const log = require(`${libs}/log`)(module); const app = require(`${libs}/app`); +// Set default port: first environment variable PORT, then configuration and last 3000 +app.set('port', process.env.PORT || config.port || 3000); process.env.NODE_ENV = process.env.NODE_ENV || 'development'; -app.set('port', process.env.PORT || config.get('port') || 3000); +app.set('port', process.env.PORT || config.port || 3000); // Set default ip: first environment variable IOP, then configuration and last '127.0.0.1' -app.set('ip', process.env.IP || config.get('ip') || '127.0.0.1'); +app.set('ip', process.env.IP || config.ip || '127.0.0.1'); const server = app.listen(app.get('port'), () => { log.info(`Express server listening on port ${server.address().port}`); diff --git a/src/test/auth.js b/src/test/auth.js new file mode 100644 index 0000000000000000000000000000000000000000..67b9d186230e80828f88c85a6fbc53489f5e4acf --- /dev/null +++ b/src/test/auth.js @@ -0,0 +1,228 @@ +process.env.NODE_ENV = 'test'; + +const chai = require('chai'); + +const dirtyChai = require('dirty-chai'); + +chai.use(dirtyChai); + +const chaiXml = require('chai-xml'); + +chai.use(chaiXml); + +const chaiHttp = require('chai-http'); + +const assert = chai.assert; + +const expect = chai.expect; + +const should = chai.should(); // actually call the function + +const libs = `${process.cwd()}/libs`; + +const server = require(`${libs}/app`); + +const mongoose = require('../libs/db/mongoose'); +//const Simulation = require('../libs/models/simulation'); +const User = require('../libs/models/user'); + +chai.use(chaiHttp); +describe('Authenticates a user', () => { + + beforeEach(() => { + User.remove({}, (err) => { + console.log('Test collection purged'); + }); + }); + + it('should authenticate a user', (done) => { + let newUser = new User(); + + newUser.email = 'lorem@ipsum.com'; + newUser.password = '123mudar'; + newUser.name = 'Gute'; + newUser.cpf = '08236017907'; + newUser.schooling = 'Doutorado'; + newUser.course = 'Ciência da Computação'; + newUser.segment = 'Comunidade acadêmica'; + newUser.role = 'Pesquisador'; + newUser.institution_name = 'UFPR'; + newUser.state = 'PR'; + newUser.city = 'Curitiba'; + + newUser.save((err) => { + if (err) { + console.log('MongoDB error:' + err); + } + }).then(function(newuser){ + chai.request(server) + .post('/api/v1/user/authenticate') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar'}) + .end((err, res) => { + let token; + + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(true); + res.body.should.have.property('token'); + token = res.body.token; + token.substr(0, 3).should.equal('JWT'); + done(); + }); + }); + }); + + it('should not authenticate a user with wrong password', (done) => { + let newUser = new User(); + + newUser.email = 'lorem@ipsum.com'; + newUser.password = '123mudar'; + newUser.name = 'Gute'; + newUser.cpf = '08236017907'; + newUser.schooling = 'Doutorado'; + newUser.course = 'Ciência da Computação'; + newUser.segment = 'Comunidade acadêmica'; + newUser.role = 'Pesquisador'; + newUser.institution_name = 'UFPR'; + newUser.state = 'PR'; + newUser.city = 'Curitiba'; + + newUser.save((err) => { + if (err) { + console.log('MongoDB error:' + err); + } + }).then(function(newuser){ + chai.request(server) + .post('/api/v1/user/authenticate') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: 'umasenhaerrada'}) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('A Senha informada é inválida.') + done(); + }); + }); + }); + + it('should not authenticate a user with wrong email', (done) => { + let newUser = new User(); + + newUser.email = 'lorem@ipsum.com'; + newUser.password = '123mudar'; + newUser.name = 'Gute'; + newUser.cpf = '08236017907'; + newUser.schooling = 'Doutorado'; + newUser.course = 'Ciência da Computação'; + newUser.segment = 'Comunidade acadêmica'; + newUser.role = 'Pesquisador'; + newUser.institution_name = 'UFPR'; + newUser.state = 'PR'; + newUser.city = 'Curitiba'; + + newUser.save((err) => { + if (err) { + console.log('MongoDB error:' + err); + } + }).then(function(newuser){ + chai.request(server) + .post('/api/v1/user/authenticate') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'dolor@ipsum.com', + password: '123mudar'}) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O Email informado não está cadastrado.') + done(); + }); + }); + }); + + it('should not authenticate a user with missing email', (done) => { + let newUser = new User(); + + newUser.email = 'lorem@ipsum.com'; + newUser.password = '123mudar'; + newUser.name = 'Gute'; + newUser.cpf = '08236017907'; + newUser.schooling = 'Doutorado'; + newUser.course = 'Ciência da Computação'; + newUser.segment = 'Comunidade acadêmica'; + newUser.role = 'Pesquisador'; + newUser.institution_name = 'UFPR'; + newUser.state = 'PR'; + newUser.city = 'Curitiba'; + + newUser.save((err) => { + if (err) { + console.log('MongoDB error:' + err); + } + }).then(function(newuser){ + chai.request(server) + .post('/api/v1/user/authenticate') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({password: '123mudar'}) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Email é obrigatório.') + done(); + }); + }); + }); + + it('should not authenticate a user with missing password', (done) => { + let newUser = new User(); + + newUser.email = 'lorem@ipsum.com'; + newUser.password = '123mudar'; + newUser.name = 'Gute'; + newUser.cpf = '08236017907'; + newUser.schooling = 'Doutorado'; + newUser.course = 'Ciência da Computação'; + newUser.segment = 'Comunidade acadêmica'; + newUser.role = 'Pesquisador'; + newUser.institution_name = 'UFPR'; + newUser.state = 'PR'; + newUser.city = 'Curitiba'; + + newUser.save((err) => { + if (err) { + console.log('MongoDB error:' + err); + } + }).then(function(newuser){ + chai.request(server) + .post('/api/v1/user/authenticate') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email:'lorem@ipsum.com'}) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Senha é obrigatório.') + done(); + }); + }); + }); +}); diff --git a/src/test/simulation.js b/src/test/simulation.js new file mode 100644 index 0000000000000000000000000000000000000000..a9349256b29bc20068eb74b0172e15d0dbe28937 --- /dev/null +++ b/src/test/simulation.js @@ -0,0 +1,869 @@ +process.env.NODE_ENV = 'test'; + +const chai = require('chai'); + +const dirtyChai = require('dirty-chai'); + +chai.use(dirtyChai); + +const chaiXml = require('chai-xml'); + +chai.use(chaiXml); + +const chaiHttp = require('chai-http'); + +const assert = chai.assert; + +const expect = chai.expect; + +const should = chai.should(); // actually call the function + +const libs = `${process.cwd()}/libs`; + +const server = require(`${libs}/app`); + +const mongoose = require('../libs/db/mongoose'); +const Simulation = require('../libs/models/simulation'); +const User = require('../libs/models/user'); + +chai.use(chaiHttp); +describe('Requires a simulation', () => { + let newSimulation; + + beforeEach(() => { + Simulation.remove({}, (err) => { + console.log('Test collection purged'); + }); + }); + + it('should create a new simulation', (done) => { + chai.request(server) + .post('/api/v1/simulation') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({ name: 'test_entry' }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('id'); + res.body.id.should.be.a('string'); + Simulation.findById(res.body.id, (err, simulation) => { + simulation.should.have.property('name'); + simulation.name.should.be.a('string'); + simulation.name.should.equal('test_entry'); + done(); + }); + }); + }); + it('should not create a nameless simulation', (done) => { + chai.request(server) + .post('/api/v1/simulation') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.not.have.property('id'); + res.body.should.have.property('success'); + res.body.success.should.equal(false); + Simulation.findById(res.body.id, (err, simulation) => { + expect(simulation).to.not.exist; + done(); + }); + }); + }); + it('should find an existing simulation', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .get(`/api/v1/simulation/${id}`) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('_id'); + res.body._id.should.be.a('string'); + res.body.should.have.property('name'); + res.body._id.should.be.a('string'); + done(); + }); + }); + }); + it('should not find an unexisting simulation', (done) => { + newSimulation = new Simulation(); + let id = newSimulation._id; + chai.request(server) + .get(`/api/v1/simulation/${id}`) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + done(); + }); + }); + it('should update an existing simulation\'s location', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ location: 5 }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('id'); + res.body.id.should.be.a('string'); + Simulation.findById(res.body.id, (err, simulation) => { + simulation.should.have.property('name'); + simulation.name.should.be.a('string'); + simulation.name.should.equal('test'); + simulation.should.have.property('location'); + simulation.location.should.be.a('number'); + simulation.location.should.equal(5); + done(); + }); + }); + }); + }); + it('should update multiple fields on a single request', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ + name: 'new_name', + location: 5, + time: 3, + failure_rate: [0.1, 0.2, 0.3], + goals_care: [0.3, 0.2, 0.1], + goals_inclusion: [0.8, 0.9, 1] + }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('id'); + res.body.id.should.be.a('string'); + Simulation.findById(res.body.id, (err, simulation) => { + simulation.should.have.property('name'); + simulation.name.should.be.a('string'); + simulation.name.should.equal('new_name'); + simulation.should.have.property('location'); + simulation.location.should.be.a('number'); + simulation.location.should.equal(5); + simulation.should.have.property('time'); + simulation.time.should.be.a('number'); + simulation.time.should.equal(3); + simulation.should.have.property('failure_rate'); + simulation.failure_rate.should.be.a('array'); + simulation.failure_rate.length.should.equal(3); + simulation.should.have.property('goals_care'); + simulation.goals_care.should.be.a('array'); + simulation.goals_care.length.should.equal(3); + simulation.should.have.property('goals_inclusion'); + simulation.goals_inclusion.should.be.a('array'); + simulation.goals_inclusion.length.should.equal(3); + done(); + }); + }); + }); + }); + it('should not update an unexisting simulation', (done) => { + newSimulation = new Simulation(); + let id = newSimulation._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ location: 5 }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + done(); + }); + }); + it('should update an existing simulation\'s time', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ time: 5 }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('id'); + res.body.id.should.be.a('string'); + Simulation.findById(res.body.id, (err, simulation) => { + simulation.should.have.property('name'); + simulation.name.should.be.a('string'); + simulation.should.have.property('time'); + simulation.time.should.be.a('number'); + simulation.time.should.equal(5); + done(); + }); + }); + }); + }); + it('should not change results for empty post requests', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + done(); + }); + }); + }); + it('should not update in case of invalid field', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ + name: 'other_name', + totally_not_valid_value_for_an_entry: 'not hacking this api', + }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + Simulation.findById(id, (err, simulation) => { + simulation.name.should.equal('test'); + done(); + }); + }); + }); + }); + it('should include consistent enrollment tables', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ + time: 5, + enrollments: "[[100, 150, 200, 250, 300]]", + }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('id'); + res.body.id.should.be.a('string'); + Simulation.findById(res.body.id, (err, simulation) => { + simulation.should.have.property('name'); + simulation.name.should.be.a('string'); + simulation.should.have.property('time'); + simulation.time.should.be.a('number'); + simulation.time.should.equal(5); + done(); + }); + }); + }); + }); + it('should not accept an invalid time', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ + time: "I'm an inocent time entry, don't mind me", + }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + }); + done(); + }); + }); + it('should not accept enrollments table different than provided time', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ + time: 5, + enrollments: "[[1,2,3]]", + }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + done(); + }); + }); + }); + it('should not include arrays of non arrays as enrollments', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ + time: 5, + enrollments: "[\"Tomato\"]", + }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + done(); + }); + }); + }); + it('should not accept non array enrollments', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ + time: 5, + enrollments: "Am I still wanted here?", + }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + done(); + }); + }); + }); + it('should not accept an enrollment with anything other than a number', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .post(`/api/v1/simulation/${id}`) + .send({ + time: 5, + enrollments: "[[1,2,\"malicious payload\",4,5]]", + }) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + done(); + }); + }); + }); + it('should delete an entry', (done) => { + newSimulation = new Simulation(); + newSimulation.name = 'test'; + newSimulation.save((err, sim) => { + let id = sim._id; + chai.request(server) + .delete(`/api/v1/simulation/${id}`) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(true); + done(); + }); + }); + }); + it('should not delete an unexisting entry', (done) => { + let sim = new Simulation(); + let id = sim._id; + chai.request(server) + .delete(`/api/v1/simulation/${id}`) + .end((err, res) => { + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + done(); + }); + }); +}); + +describe('Saves a user', () => { + + beforeEach(() => { + User.remove({}, (err) => { + console.log('Test collection purged'); + }); + }); + + it('should create a new user', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila Baby', + cpf: '48303270737', + schooling: 'Doutorado', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(true); + User.findOne({'email': 'lorem@ipsum.com'}, (err, user) => { + if (err){ + console.log('MongoDB error: ' + err); + } + + user.should.have.property('email'); + done(); + }); + }); + }); + + it('should not create a user with an email that is already in use', (done) => { + let newUser = new User(); + + newUser.email = 'lorem@ipsum.com'; + newUser.password = '123mudar'; + newUser.name = 'Gute'; + newUser.cpf = '08236017907'; + newUser.schooling = 'Doutorado'; + newUser.course = 'Ciência da Computação'; + newUser.segment = 'Comunidade acadêmica'; + newUser.role = 'Pesquisador'; + newUser.institution_name = 'UFPR'; + newUser.state = 'PR'; + newUser.city = 'Curitiba'; + + newUser.save((err) => { + if (err) { + console.log('MongoDB error:' + err); + } + }).then(function(newuser){ + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila Baby', + cpf: '48303270737', + schooling: 'Doutorado', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O email informado já está cadastrado.'); + User.findOne({'cpf': '48303270737'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + }); + + it('should not create a user with a CPF that is already in use', (done) => { + let newUser = new User(); + + newUser.email = 'lorem@ipsum.com'; + newUser.password = '123mudar'; + newUser.name = 'Gute'; + newUser.cpf = '08236017907'; + newUser.schooling = 'Doutorado'; + newUser.course = 'Ciência da Computação'; + newUser.segment = 'Comunidade acadêmica'; + newUser.role = 'Pesquisador'; + newUser.institution_name = 'UFPR'; + newUser.state = 'PR'; + newUser.city = 'Curitiba'; + + newUser.save((err) => { + if (err) { + console.log('MongoDB error:' + err); + } + }).then(function(newuser){ + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'dolor@ipsum.com', + password: '123mudar', + name: 'Tequila Baby', + cpf: '08236017907', + schooling: 'Doutorado', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O CPF informado já está cadastrado.'); + User.findOne({'email': 'dolor@ipsum.com'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + }); + + it('should not save an user without email', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({password: '123mudar', + name: 'Tequila Baby', + cpf: '48303270737', + schooling: 'Doutorado', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Email é obrigatório.'); + User.findOne({'cpf': '48303270737'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without password', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + name: 'Tequila Baby', + cpf: '48303270737', + schooling: 'Doutorado', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Senha é obrigatório.'); + User.findOne({'cpf': '48303270737'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user with invalid email', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'notavalidemail', + password: '123mudar', + name: 'Tequila Baby', + cpf: '48303270737', + schooling: 'Doutorado', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O email informado é inválido.'); + User.findOne({'cpf': '48303270737'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without name', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + cpf: '48303270737', + schooling: 'Doutorado', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Nome é obrigatório.'); + User.findOne({'cpf': '48303270737'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without CPF', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila baby', + schooling: 'Doutorado', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo CPF é obrigatório.'); + User.findOne({'email': 'lorem@ipsum.com'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without segment', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila baby', + schooling: 'Doutorado', + cpf: '48303270737', + course: 'Ciência da Computação', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Segmento é obrigatório.'); + User.findOne({'email': 'lorem@ipsum.com'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without schooling', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila baby', + cpf: '48303270737', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Escolaridade é obrigatório.'); + User.findOne({'email': 'lorem@ipsum.com'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without role', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila baby', + schooling: 'Doutorado', + cpf: '48303270737', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + institution_name: 'UFPR', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Função é obrigatório.'); + User.findOne({'email': 'lorem@ipsum.com'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without institution', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila baby', + schooling: 'Doutorado', + cpf: '48303270737', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + role: 'Pesquisador', + state: 'PR', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Instituição em que trabalha é obrigatório.'); + User.findOne({'email': 'lorem@ipsum.com'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without city', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila baby', + schooling: 'Doutorado', + cpf: '48303270737', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + institution_name: 'UFPR', + role: 'Pesquisador', + state: 'PR'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Cidade é obrigatório.'); + User.findOne({'email': 'lorem@ipsum.com'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + + it('should not save an user without state', (done) => { + chai.request(server) + .post('/api/v1/user') + .set('content-type', 'application/x-www-form-urlencoded') + .set('x-apicache-bypass', 'true') + .send({email: 'lorem@ipsum.com', + password: '123mudar', + name: 'Tequila baby', + schooling: 'Doutorado', + cpf: '48303270737', + course: 'Ciência da Computação', + segment: 'Comunidade acadêmica', + institution_name: 'UFPR', + role: 'Pesquisador', + city: 'Cutiriba'}) + .end((err, res) =>{ + res.should.have.status(200); + res.should.be.json; + res.body.should.have.property('success'); + res.body.success.should.equal(false); + res.body.should.have.property('msg'); + res.body.msg.should.equal('O campo Estado é obrigatório.'); + User.findOne({'email': 'lorem@ipsum.com'}, (err, user) => { + expect(user).to.not.exist; + done(); + }); + }); + }); + +});