overleaf/services/web/app/coffee/Features/User/UserGetter.coffee

107 lines
3.5 KiB
CoffeeScript
Raw Normal View History

2014-02-12 10:23:40 +00:00
mongojs = require("../../infrastructure/mongojs")
metrics = require('metrics-sharelatex')
2017-03-16 10:59:18 +00:00
logger = require('logger-sharelatex')
2014-02-12 10:23:40 +00:00
db = mongojs.db
ObjectId = mongojs.ObjectId
2018-06-27 16:29:56 +00:00
{ getAffiliations } = require("./UserAffiliationsManager")
2018-07-17 10:12:09 +00:00
Errors = require("../Errors/Errors")
2014-02-12 10:23:40 +00:00
module.exports = UserGetter =
getUser: (query, projection, callback = (error, user) ->) ->
if query?.email?
return callback(new Error("Don't use getUser to find user by email"), null)
2014-02-12 10:23:40 +00:00
if arguments.length == 2
callback = projection
projection = {}
if typeof query == "string"
2014-11-09 23:08:23 +00:00
try
query = _id: ObjectId(query)
catch e
return callback(null, null)
2014-02-12 10:23:40 +00:00
else if query instanceof ObjectId
query = _id: query
db.users.findOne query, projection, callback
2018-05-25 11:04:09 +00:00
getUserEmail: (userId, callback = (error, email) ->) ->
@getUser userId, { email: 1 }, (error, user) ->
callback(error, user?.email)
2018-06-08 17:05:19 +00:00
getUserFullEmails: (userId, callback = (error, emails) ->) ->
@getUser userId, { email: 1, emails: 1 }, (error, user) ->
return callback error if error?
return callback new Error('User not Found') unless user
getAffiliations userId, (error, affiliationsData) ->
return callback error if error?
callback null, decorateFullEmails(user.email, user.emails, affiliationsData)
2018-06-08 17:05:19 +00:00
getUserByMainEmail: (email, projection, callback = (error, user) ->) ->
email = email.trim()
if arguments.length == 2
callback = projection
projection = {}
db.users.findOne email: email, projection, callback
2018-05-25 11:04:09 +00:00
getUserByAnyEmail: (email, projection, callback = (error, user) ->) ->
email = email.trim()
if arguments.length == 2
callback = projection
projection = {}
2018-06-06 13:46:41 +00:00
# $exists: true MUST be set to use the partial index
query = emails: { $exists: true }, 'emails.email': email
db.users.findOne query, projection, (error, user) =>
2018-05-25 11:04:09 +00:00
return callback(error, user) if error? or user?
# While multiple emails are being rolled out, check for the main email as
# well
@getUserByMainEmail email, projection, callback
getUsers: (user_ids, projection, callback = (error, users) ->) ->
try
user_ids = user_ids.map (u) -> ObjectId(u.toString())
catch error
return callback error
2017-03-16 10:59:18 +00:00
db.users.find { _id: { $in: user_ids} }, projection, callback
getUserOrUserStubById: (user_id, projection, callback = (error, user) ->) ->
try
query = _id: ObjectId(user_id.toString())
catch e
return callback(new Error(e))
db.users.findOne query, projection, (error, user) ->
return callback(error) if error?
return callback(null, user) if user?
db.userstubs.findOne query, projection, callback
2017-03-17 14:55:41 +00:00
2018-05-28 14:08:37 +00:00
# check for duplicate email address. This is also enforced at the DB level
ensureUniqueEmailAddress: (newEmail, callback) ->
@getUserByAnyEmail newEmail, (error, user) ->
2018-07-17 10:12:09 +00:00
return callback(new Errors.EmailExistsError('alread_exists')) if user?
2018-05-28 14:08:37 +00:00
callback(error)
decorateFullEmails = (defaultEmail, emailsData, affiliationsData) ->
emailsData.map (emailData) ->
emailData.default = emailData.email == defaultEmail
affiliation = affiliationsData.find (aff) -> aff.email == emailData.email
if affiliation?
{ institution, inferred, role, department } = affiliation
emailData.affiliation = { institution, inferred, role, department }
else
emailsData.affiliation = null
emailData
2017-03-17 14:55:41 +00:00
[
'getUser',
2018-05-25 11:04:09 +00:00
'getUserEmail',
'getUserByMainEmail',
2018-05-25 11:04:09 +00:00
'getUserByAnyEmail',
'getUsers',
2018-05-28 14:08:37 +00:00
'getUserOrUserStubById',
'ensureUniqueEmailAddress',
2017-03-17 14:55:41 +00:00
].map (method) ->
metrics.timeAsyncMethod UserGetter, method, 'mongo.UserGetter', logger