2014-05-19 06:50:32 -04:00
|
|
|
logger = require("logger-sharelatex")
|
2014-02-12 05:23:40 -05:00
|
|
|
mongojs = require("../../infrastructure/mongojs")
|
2017-04-03 11:18:30 -04:00
|
|
|
metrics = require("metrics-sharelatex")
|
2014-02-12 05:23:40 -05:00
|
|
|
db = mongojs.db
|
2018-05-25 07:04:09 -04:00
|
|
|
async = require("async")
|
2014-02-12 05:23:40 -05:00
|
|
|
ObjectId = mongojs.ObjectId
|
2018-05-23 10:12:23 -04:00
|
|
|
UserGetter = require("./UserGetter")
|
2018-06-19 08:55:34 -04:00
|
|
|
EmailHelper = require "../Helpers/EmailHelper"
|
|
|
|
Errors = require "../Errors/Errors"
|
2018-06-19 07:46:15 -04:00
|
|
|
settings = require "settings-sharelatex"
|
|
|
|
request = require "request"
|
2014-02-12 05:23:40 -05:00
|
|
|
|
|
|
|
module.exports = UserUpdater =
|
|
|
|
updateUser: (query, update, callback = (error) ->) ->
|
|
|
|
if typeof query == "string"
|
|
|
|
query = _id: ObjectId(query)
|
|
|
|
else if query instanceof ObjectId
|
|
|
|
query = _id: query
|
2018-05-25 07:04:09 -04:00
|
|
|
else if typeof query._id == "string"
|
|
|
|
query._id = ObjectId(query._id)
|
2014-02-12 05:23:40 -05:00
|
|
|
|
|
|
|
db.users.update query, update, callback
|
2014-05-16 12:29:54 -04:00
|
|
|
|
|
|
|
|
2018-05-25 07:04:09 -04:00
|
|
|
#
|
|
|
|
# DEPRECATED
|
|
|
|
#
|
|
|
|
# Change the user's main email address by adding a new email, switching the
|
|
|
|
# default email and removing the old email. Prefer manipulating multiple
|
|
|
|
# emails and the default rather than calling this method directly
|
|
|
|
#
|
|
|
|
changeEmailAddress: (userId, newEmail, callback)->
|
|
|
|
logger.log userId: userId, newEmail: newEmail, "updaing email address of user"
|
|
|
|
|
|
|
|
oldEmail = null
|
|
|
|
async.series [
|
|
|
|
(cb) ->
|
|
|
|
UserGetter.getUserEmail userId, (error, email) ->
|
|
|
|
oldEmail = email
|
|
|
|
cb(error)
|
|
|
|
(cb) -> UserUpdater.addEmailAddress userId, newEmail, cb
|
|
|
|
(cb) -> UserUpdater.setDefaultEmailAddress userId, newEmail, cb
|
|
|
|
(cb) -> UserUpdater.removeEmailAddress userId, oldEmail, cb
|
|
|
|
], callback
|
|
|
|
|
|
|
|
|
|
|
|
# Add a new email address for the user. Email cannot be already used by this
|
|
|
|
# or any other user
|
2018-06-19 07:46:15 -04:00
|
|
|
addEmailAddress: (userId, newEmail, affiliationOptions, callback) ->
|
|
|
|
unless callback? # affiliationOptions is optional
|
|
|
|
callback = affiliationOptions
|
|
|
|
affiliationOptions = {}
|
|
|
|
|
2018-05-28 10:08:37 -04:00
|
|
|
UserGetter.ensureUniqueEmailAddress newEmail, (error) =>
|
2018-05-25 07:04:09 -04:00
|
|
|
return callback(error) if error?
|
|
|
|
|
2018-06-19 07:46:15 -04:00
|
|
|
addAffiliation userId, newEmail, affiliationOptions, (error) =>
|
2018-05-25 07:04:09 -04:00
|
|
|
if error?
|
2018-06-19 07:46:15 -04:00
|
|
|
logger.err error: error, 'problem adding affiliation'
|
2018-05-25 07:04:09 -04:00
|
|
|
return callback(error)
|
2018-06-19 07:46:15 -04:00
|
|
|
|
|
|
|
update = $push: emails: email: newEmail, createdAt: new Date()
|
|
|
|
@updateUser userId, update, (error) ->
|
|
|
|
if error?
|
|
|
|
logger.err error: error, 'problem updating users emails'
|
|
|
|
return callback(error)
|
|
|
|
callback()
|
2014-05-16 12:29:54 -04:00
|
|
|
|
2018-05-25 07:04:09 -04:00
|
|
|
# remove one of the user's email addresses. The email cannot be the user's
|
|
|
|
# default email address
|
|
|
|
removeEmailAddress: (userId, email, callback) ->
|
2018-06-19 07:46:15 -04:00
|
|
|
removeAffiliation userId, email, (error) =>
|
2018-05-25 07:04:09 -04:00
|
|
|
if error?
|
2018-06-19 07:46:15 -04:00
|
|
|
logger.err error: error, 'problem removing affiliation'
|
2018-05-25 07:04:09 -04:00
|
|
|
return callback(error)
|
2018-06-19 07:46:15 -04:00
|
|
|
|
|
|
|
query = _id: userId, email: $ne: email
|
|
|
|
update = $pull: emails: email: email
|
|
|
|
@updateUser query, update, (error, res) ->
|
|
|
|
if error?
|
|
|
|
logger.err error:error, 'problem removing users email'
|
|
|
|
return callback(error)
|
|
|
|
if res.n == 0
|
|
|
|
return callback(new Error('Cannot remove default email'))
|
|
|
|
callback()
|
2018-05-25 07:04:09 -04:00
|
|
|
|
|
|
|
|
|
|
|
# set the default email address by setting the `email` attribute. The email
|
|
|
|
# must be one of the user's multiple emails (`emails` attribute)
|
|
|
|
setDefaultEmailAddress: (userId, email, callback) ->
|
|
|
|
query = _id: userId, 'emails.email': email
|
|
|
|
update = $set: email: email
|
|
|
|
@updateUser query, update, (error, res) ->
|
|
|
|
if error?
|
|
|
|
logger.err error:error, 'problem setting default emails'
|
|
|
|
return callback(error)
|
2018-06-19 08:55:34 -04:00
|
|
|
if res.n == 0 # TODO: Check n or nMatched?
|
2018-05-25 07:04:09 -04:00
|
|
|
return callback(new Error('Default email does not belong to user'))
|
|
|
|
callback()
|
|
|
|
|
2018-06-19 08:55:34 -04:00
|
|
|
confirmEmail: (userId, email, callback) ->
|
|
|
|
email = EmailHelper.parseEmail(email)
|
|
|
|
return callback(new Error('invalid email')) if !email?
|
|
|
|
logger.log {userId, email}, 'confirming user email'
|
|
|
|
query =
|
|
|
|
_id: userId
|
|
|
|
'emails.email': email
|
|
|
|
update =
|
|
|
|
$set:
|
|
|
|
'emails.$.confirmedAt': new Date()
|
|
|
|
@updateUser query, update, (error, res) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
logger.log {res, userId, email}, "tried to confirm email"
|
|
|
|
if res.n == 0
|
|
|
|
return callback(new Errors.NotFoundError('user id and email do no match'))
|
|
|
|
callback()
|
|
|
|
|
2018-06-19 07:46:15 -04:00
|
|
|
addAffiliation = (userId, email, { university, department, role }, callback = (error) ->) ->
|
|
|
|
makeAffiliationRequest {
|
|
|
|
method: 'POST'
|
|
|
|
path: "/api/v2/users/#{userId.toString()}/affiliations"
|
|
|
|
body: { email, university, department, role }
|
|
|
|
defaultErrorMessage: "Couldn't create affiliation"
|
|
|
|
}, callback
|
|
|
|
|
|
|
|
removeAffiliation = (userId, email, callback = (error) ->) ->
|
|
|
|
email = encodeURIComponent(email)
|
|
|
|
makeAffiliationRequest {
|
|
|
|
method: 'DELETE'
|
|
|
|
path: "/api/v2/users/#{userId.toString()}/affiliations/#{email}"
|
|
|
|
extraSuccessStatusCodes: [404] # `Not Found` responses are considered successful
|
|
|
|
defaultErrorMessage: "Couldn't remove affiliation"
|
|
|
|
}, callback
|
|
|
|
|
2018-06-21 07:31:55 -04:00
|
|
|
makeAffiliationRequest = (requestOptions, callback = (error) ->) ->
|
|
|
|
return callback(null) unless settings?.apis?.v1?.url # service is not configured
|
2018-06-19 07:46:15 -04:00
|
|
|
requestOptions.extraSuccessStatusCodes ||= []
|
|
|
|
request {
|
|
|
|
method: requestOptions.method
|
|
|
|
url: "#{settings.apis.v1.url}#{requestOptions.path}"
|
|
|
|
body: requestOptions.body
|
|
|
|
auth: { user: settings.apis.v1.user, pass: settings.apis.v1.pass }
|
|
|
|
json: true,
|
|
|
|
timeout: 20 * 1000
|
|
|
|
}, (error, response, body) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
isSuccess = 200 <= response.statusCode < 300
|
|
|
|
isSuccess ||= response.statusCode in requestOptions.extraSuccessStatusCodes
|
|
|
|
unless isSuccess
|
|
|
|
if body?.errors
|
|
|
|
errorMessage = "#{response.statusCode}: #{body.errors}"
|
|
|
|
else
|
|
|
|
errorMessage = "#{requestOptions.defaultErrorMessage}: #{response.statusCode}"
|
|
|
|
return callback(new Error(errorMessage))
|
|
|
|
|
|
|
|
callback(null)
|
2018-06-19 08:55:34 -04:00
|
|
|
|
2018-05-25 07:04:09 -04:00
|
|
|
[
|
|
|
|
'updateUser'
|
|
|
|
'changeEmailAddress'
|
|
|
|
'setDefaultEmailAddress'
|
|
|
|
'addEmailAddress'
|
|
|
|
'removeEmailAddress'
|
|
|
|
].map (method) ->
|
|
|
|
metrics.timeAsyncMethod(UserUpdater, method, 'mongo.UserUpdater', logger)
|