mirror of
https://github.com/overleaf/overleaf.git
synced 2024-10-31 21:21:03 -04:00
d3a30929f7
GitOrigin-RevId: a58ecfb13c25df02ccf79c189903b5a6fcddd835
220 lines
5.6 KiB
JavaScript
220 lines
5.6 KiB
JavaScript
const mongojs = require('../../infrastructure/mongojs')
|
|
const metrics = require('metrics-sharelatex')
|
|
const logger = require('logger-sharelatex')
|
|
const { db } = mongojs
|
|
const { ObjectId } = mongojs
|
|
const { promisifyAll } = require('../../util/promises')
|
|
const { getUserAffiliations } = require('../Institutions/InstitutionsAPI')
|
|
const InstitutionsHelper = require('../Institutions/InstitutionsHelper')
|
|
const Errors = require('../Errors/Errors')
|
|
const Features = require('../../infrastructure/Features')
|
|
|
|
const UserGetter = {
|
|
getUser(query, projection, callback) {
|
|
if (arguments.length === 2) {
|
|
callback = projection
|
|
projection = {}
|
|
}
|
|
try {
|
|
query = normalizeQuery(query)
|
|
db.users.findOne(query, projection, callback)
|
|
} catch (err) {
|
|
callback(err)
|
|
}
|
|
},
|
|
|
|
getUserEmail(userId, callback) {
|
|
this.getUser(userId, { email: 1 }, (error, user) =>
|
|
callback(error, user && user.email)
|
|
)
|
|
},
|
|
|
|
getUserFullEmails(userId, callback) {
|
|
this.getUser(userId, { email: 1, emails: 1, samlIdentifiers: 1 }, function(
|
|
error,
|
|
user
|
|
) {
|
|
if (error) {
|
|
return callback(error)
|
|
}
|
|
if (!user) {
|
|
return callback(new Error('User not Found'))
|
|
}
|
|
|
|
if (!Features.hasFeature('affiliations')) {
|
|
return callback(
|
|
null,
|
|
decorateFullEmails(user.email, user.emails, [], [])
|
|
)
|
|
}
|
|
|
|
getUserAffiliations(userId, function(error, affiliationsData) {
|
|
if (error) {
|
|
return callback(error)
|
|
}
|
|
callback(
|
|
null,
|
|
decorateFullEmails(
|
|
user.email,
|
|
user.emails || [],
|
|
affiliationsData,
|
|
user.samlIdentifiers || []
|
|
)
|
|
)
|
|
})
|
|
})
|
|
},
|
|
|
|
getUserByMainEmail(email, projection, callback) {
|
|
email = email.trim()
|
|
if (arguments.length === 2) {
|
|
callback = projection
|
|
projection = {}
|
|
}
|
|
db.users.findOne({ email }, projection, callback)
|
|
},
|
|
|
|
getUserByAnyEmail(email, projection, callback) {
|
|
email = email.trim()
|
|
if (arguments.length === 2) {
|
|
callback = projection
|
|
projection = {}
|
|
}
|
|
// $exists: true MUST be set to use the partial index
|
|
const query = { emails: { $exists: true }, 'emails.email': email }
|
|
db.users.findOne(query, projection, (error, user) => {
|
|
if (error || user) {
|
|
return callback(error, user)
|
|
}
|
|
|
|
// While multiple emails are being rolled out, check for the main email as
|
|
// well
|
|
this.getUserByMainEmail(email, projection, callback)
|
|
})
|
|
},
|
|
|
|
getUsersByAnyConfirmedEmail(emails, projection, callback) {
|
|
if (arguments.length === 2) {
|
|
callback = projection
|
|
projection = {}
|
|
}
|
|
// $exists: true MUST be set to use the partial index
|
|
const query = {
|
|
emails: {
|
|
$exists: true,
|
|
$elemMatch: { email: { $in: emails }, confirmedAt: { $exists: true } }
|
|
}
|
|
}
|
|
db.users.find(query, projection, callback)
|
|
},
|
|
|
|
getUsersByV1Ids(v1Ids, projection, callback) {
|
|
if (arguments.length === 2) {
|
|
callback = projection
|
|
projection = {}
|
|
}
|
|
const query = { 'overleaf.id': { $in: v1Ids } }
|
|
db.users.find(query, projection, callback)
|
|
},
|
|
|
|
getUsersByHostname(hostname, projection, callback) {
|
|
const reversedHostname = hostname
|
|
.trim()
|
|
.split('')
|
|
.reverse()
|
|
.join('')
|
|
const query = {
|
|
emails: { $exists: true },
|
|
'emails.reversedHostname': reversedHostname
|
|
}
|
|
db.users.find(query, projection, callback)
|
|
},
|
|
|
|
getUsers(query, projection, callback) {
|
|
try {
|
|
query = normalizeQuery(query)
|
|
db.users.find(query, projection, callback)
|
|
} catch (err) {
|
|
callback(err)
|
|
}
|
|
},
|
|
|
|
// check for duplicate email address. This is also enforced at the DB level
|
|
ensureUniqueEmailAddress(newEmail, callback) {
|
|
this.getUserByAnyEmail(newEmail, function(error, user) {
|
|
if (user) {
|
|
return callback(new Errors.EmailExistsError())
|
|
}
|
|
callback(error)
|
|
})
|
|
}
|
|
}
|
|
|
|
function normalizeQuery(query) {
|
|
if (!query) {
|
|
throw new Error('no query provided')
|
|
}
|
|
if (typeof query === 'string') {
|
|
return { _id: ObjectId(query) }
|
|
} else if (query instanceof ObjectId) {
|
|
return { _id: query }
|
|
} else if (Array.isArray(query)) {
|
|
const userIds = query.map(u => ObjectId(u.toString()))
|
|
return { _id: { $in: userIds } }
|
|
} else {
|
|
return query
|
|
}
|
|
}
|
|
|
|
var decorateFullEmails = (
|
|
defaultEmail,
|
|
emailsData,
|
|
affiliationsData,
|
|
samlIdentifiers
|
|
) =>
|
|
emailsData.map(function(emailData) {
|
|
emailData.default = emailData.email === defaultEmail
|
|
|
|
const affiliation = affiliationsData.find(
|
|
aff => aff.email === emailData.email
|
|
)
|
|
if (affiliation) {
|
|
const { institution, inferred, role, department, licence } = affiliation
|
|
emailData.affiliation = {
|
|
institution,
|
|
inferred,
|
|
role,
|
|
department,
|
|
licence
|
|
}
|
|
} else {
|
|
emailsData.affiliation = null
|
|
}
|
|
|
|
if (emailData.samlProviderId) {
|
|
emailData.samlIdentifier = samlIdentifiers.find(
|
|
samlIdentifier => samlIdentifier.providerId === emailData.samlProviderId
|
|
)
|
|
} else {
|
|
emailsData.samlIdentifier = null
|
|
}
|
|
|
|
emailData.emailHasInstitutionLicence = InstitutionsHelper.emailHasLicence(
|
|
emailData
|
|
)
|
|
|
|
return emailData
|
|
})
|
|
;[
|
|
'getUser',
|
|
'getUserEmail',
|
|
'getUserByMainEmail',
|
|
'getUserByAnyEmail',
|
|
'getUsers',
|
|
'ensureUniqueEmailAddress'
|
|
].map(method =>
|
|
metrics.timeAsyncMethod(UserGetter, method, 'mongo.UserGetter', logger)
|
|
)
|
|
|
|
UserGetter.promises = promisifyAll(UserGetter)
|
|
module.exports = UserGetter
|