2016-06-17 11:11:39 +00:00
|
|
|
Settings = require "settings-sharelatex"
|
2014-02-12 10:23:40 +00:00
|
|
|
User = require("../../models/User").User
|
|
|
|
{db, ObjectId} = require("../../infrastructure/mongojs")
|
|
|
|
crypto = require 'crypto'
|
|
|
|
bcrypt = require 'bcrypt'
|
2018-10-04 12:47:53 +00:00
|
|
|
EmailHelper = require("../Helpers/EmailHelper")
|
2014-02-12 10:23:40 +00:00
|
|
|
|
2016-06-17 11:11:39 +00:00
|
|
|
BCRYPT_ROUNDS = Settings?.security?.bcryptRounds or 12
|
|
|
|
|
2014-02-12 10:23:40 +00:00
|
|
|
module.exports = AuthenticationManager =
|
|
|
|
authenticate: (query, password, callback = (error, user) ->) ->
|
|
|
|
# Using Mongoose for legacy reasons here. The returned User instance
|
|
|
|
# gets serialized into the session and there may be subtle differences
|
|
|
|
# between the user returned by Mongoose vs mongojs (such as default values)
|
|
|
|
User.findOne query, (error, user) =>
|
|
|
|
return callback(error) if error?
|
|
|
|
if user?
|
|
|
|
if user.hashedPassword?
|
|
|
|
bcrypt.compare password, user.hashedPassword, (error, match) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
if match
|
2016-06-17 11:11:39 +00:00
|
|
|
AuthenticationManager.checkRounds user, user.hashedPassword, password, (err) ->
|
|
|
|
return callback(err) if err?
|
|
|
|
callback null, user
|
2014-02-12 10:23:40 +00:00
|
|
|
else
|
|
|
|
callback null, null
|
|
|
|
else
|
|
|
|
callback null, null
|
|
|
|
else
|
|
|
|
callback null, null
|
|
|
|
|
2018-10-04 11:00:33 +00:00
|
|
|
validateEmail: (email) ->
|
2018-10-04 12:47:53 +00:00
|
|
|
parsed = EmailHelper.parseEmail(email)
|
|
|
|
if !parsed?
|
|
|
|
return { message: 'email not valid' }
|
2018-10-04 11:00:33 +00:00
|
|
|
return null
|
|
|
|
|
|
|
|
validatePassword: (password) ->
|
|
|
|
if !password?
|
|
|
|
return { message: 'password not set' }
|
2017-07-24 10:06:47 +00:00
|
|
|
if (Settings.passwordStrengthOptions?.length?.max? and
|
2018-10-08 10:25:24 +00:00
|
|
|
password.length > Settings.passwordStrengthOptions?.length?.max)
|
2018-10-05 09:18:53 +00:00
|
|
|
return { message: "password is too long" }
|
2018-10-08 10:25:24 +00:00
|
|
|
if (Settings.passwordStrengthOptions?.length?.min? and
|
|
|
|
password.length < Settings.passwordStrengthOptions?.length?.min)
|
|
|
|
return { message: 'password is too short' }
|
2018-10-04 11:00:33 +00:00
|
|
|
return null
|
|
|
|
|
|
|
|
setUserPassword: (user_id, password, callback = (error) ->) ->
|
2018-10-04 12:47:53 +00:00
|
|
|
validation = @validatePassword(password)
|
2018-10-04 11:00:33 +00:00
|
|
|
return callback(validation.message) if validation?
|
2016-09-23 14:44:47 +00:00
|
|
|
|
2016-06-17 11:11:39 +00:00
|
|
|
bcrypt.genSalt BCRYPT_ROUNDS, (error, salt) ->
|
2014-02-12 10:23:40 +00:00
|
|
|
return callback(error) if error?
|
|
|
|
bcrypt.hash password, salt, (error, hash) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
db.users.update({
|
|
|
|
_id: ObjectId(user_id.toString())
|
|
|
|
}, {
|
|
|
|
$set: hashedPassword: hash
|
|
|
|
$unset: password: true
|
|
|
|
}, callback)
|
|
|
|
|
2016-06-17 11:11:39 +00:00
|
|
|
checkRounds: (user, hashedPassword, password, callback = (error) ->) ->
|
|
|
|
# check current number of rounds and rehash if necessary
|
|
|
|
currentRounds = bcrypt.getRounds hashedPassword
|
|
|
|
if currentRounds < BCRYPT_ROUNDS
|
|
|
|
AuthenticationManager.setUserPassword user._id, password, callback
|
|
|
|
else
|
|
|
|
callback()
|