mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-06-10 06:39:01 +02:00
394c60f2cf
Revert "Revert "[web] Convert models and self-referential test files to ESM "" GitOrigin-RevId: f64000ae31d298b075a8722dfc51f294c71bc021
373 lines
10 KiB
JavaScript
373 lines
10 KiB
JavaScript
import { callbackify } from 'node:util'
|
|
import { db } from '../../infrastructure/mongodb.js'
|
|
import moment from 'moment'
|
|
import settings from '@overleaf/settings'
|
|
import InstitutionsAPI from '../Institutions/InstitutionsAPI.mjs'
|
|
import InstitutionsHelper from '../Institutions/InstitutionsHelper.mjs'
|
|
import Errors from '../Errors/Errors.js'
|
|
import Features from '../../infrastructure/Features.js'
|
|
import { User } from '../../models/User.mjs'
|
|
import { normalizeQuery, normalizeMultiQuery } from '../Helpers/Mongo.js'
|
|
import Modules from '../../infrastructure/Modules.js'
|
|
import FeaturesHelper from '../Subscription/FeaturesHelper.mjs'
|
|
import AsyncLocalStorage from '../../infrastructure/AsyncLocalStorage.js'
|
|
|
|
const InstitutionsAPIPromises = InstitutionsAPI.promises
|
|
|
|
function _lastDayToReconfirm(emailData, institutionData) {
|
|
const globalReconfirmPeriod = settings.reconfirmNotificationDays
|
|
if (!globalReconfirmPeriod) return undefined
|
|
|
|
// only show notification for institutions with reconfirmation enabled
|
|
if (!institutionData || !institutionData.maxConfirmationMonths)
|
|
return undefined
|
|
|
|
if (!emailData.confirmedAt) return undefined
|
|
|
|
if (institutionData.ssoEnabled && !emailData.samlProviderId) {
|
|
// For SSO, only show notification for linked email
|
|
return false
|
|
}
|
|
|
|
// reconfirmedAt will not always be set, use confirmedAt as fallback
|
|
const lastConfirmed = emailData.reconfirmedAt || emailData.confirmedAt
|
|
|
|
return moment(lastConfirmed)
|
|
.add(institutionData.maxConfirmationMonths, 'months')
|
|
.toDate()
|
|
}
|
|
|
|
function _pastReconfirmDate(lastDayToReconfirm) {
|
|
if (!lastDayToReconfirm) return false
|
|
return moment(lastDayToReconfirm).isBefore()
|
|
}
|
|
|
|
function _emailInReconfirmNotificationPeriod(
|
|
cachedLastDayToReconfirm,
|
|
lastDayToReconfirm
|
|
) {
|
|
const globalReconfirmPeriod = settings.reconfirmNotificationDays
|
|
|
|
if (!globalReconfirmPeriod || !cachedLastDayToReconfirm) return false
|
|
|
|
const notificationStarts = moment(cachedLastDayToReconfirm).subtract(
|
|
globalReconfirmPeriod,
|
|
'days'
|
|
)
|
|
|
|
let isInNotificationPeriod = moment().isAfter(notificationStarts)
|
|
|
|
if (!isInNotificationPeriod) {
|
|
// for possible issues in v1/v2 date mismatch, ensure v2 date doesn't show as needing to reconfirm
|
|
|
|
const notificationStartsV2 = moment(lastDayToReconfirm).subtract(
|
|
globalReconfirmPeriod,
|
|
'days'
|
|
)
|
|
|
|
isInNotificationPeriod = moment().isAfter(notificationStartsV2)
|
|
}
|
|
|
|
return isInNotificationPeriod
|
|
}
|
|
|
|
async function getUserFullEmails(userId) {
|
|
const store = AsyncLocalStorage.storage.getStore()
|
|
if (store?.userFullEmails?.[userId]) {
|
|
return store.userFullEmails[userId]
|
|
}
|
|
const user = await UserGetter.promises.getUser(userId, {
|
|
email: 1,
|
|
emails: 1,
|
|
samlIdentifiers: 1,
|
|
})
|
|
|
|
if (!user) {
|
|
throw new Error('User not Found')
|
|
}
|
|
|
|
if (!Features.hasFeature('affiliations')) {
|
|
return decorateFullEmails(user.email, user.emails, [], [])
|
|
}
|
|
|
|
const affiliationsData =
|
|
await InstitutionsAPIPromises.getUserAffiliations(userId)
|
|
|
|
const fullEmails = decorateFullEmails(
|
|
user.email,
|
|
user.emails || [],
|
|
affiliationsData,
|
|
user.samlIdentifiers || []
|
|
)
|
|
|
|
if (store) {
|
|
if (!store.userFullEmails) {
|
|
store.userFullEmails = {}
|
|
}
|
|
store.userFullEmails[userId] = fullEmails
|
|
}
|
|
return fullEmails
|
|
}
|
|
|
|
async function getUserFeatures(userId) {
|
|
const user = await UserGetter.promises.getUser(userId, {
|
|
features: 1,
|
|
})
|
|
if (!user) {
|
|
throw new Error('User not Found')
|
|
}
|
|
|
|
const moduleFeatures =
|
|
(await Modules.promises.hooks.fire('getModuleProvidedFeatures', userId)) ||
|
|
[]
|
|
|
|
return FeaturesHelper.computeFeatureSet([user.features, ...moduleFeatures])
|
|
}
|
|
|
|
async function getUserConfirmedEmails(userId) {
|
|
const user = await UserGetter.promises.getUser(userId, {
|
|
emails: 1,
|
|
})
|
|
|
|
if (!user) {
|
|
throw new Error('User not Found')
|
|
}
|
|
|
|
return user.emails.filter(email => !!email.confirmedAt)
|
|
}
|
|
|
|
async function getSsoUsersAtInstitution(institutionId, projection) {
|
|
if (!projection) {
|
|
throw new Error('missing projection')
|
|
}
|
|
|
|
return await User.find(
|
|
{
|
|
'samlIdentifiers.providerId': institutionId.toString(),
|
|
},
|
|
projection
|
|
).exec()
|
|
}
|
|
|
|
async function getWritefullData(userId) {
|
|
const user = await UserGetter.promises.getUser(userId, {
|
|
writefull: 1,
|
|
})
|
|
if (!user) {
|
|
throw new Error('user not found')
|
|
}
|
|
return {
|
|
isPremium: Boolean(user?.writefull?.isPremium),
|
|
premiumSource: user?.writefull?.premiumSource || null,
|
|
}
|
|
}
|
|
|
|
async function getUser(query, projection = {}) {
|
|
query = normalizeQuery(query)
|
|
return await db.users.findOne(query, { projection })
|
|
}
|
|
|
|
async function getUserEmail(userId) {
|
|
const user = await UserGetter.promises.getUser(userId, { email: 1 })
|
|
return user && user.email
|
|
}
|
|
|
|
async function getUserByMainEmail(email, projection = {}) {
|
|
email = email.trim()
|
|
return await db.users.findOne({ email }, { projection })
|
|
}
|
|
|
|
async function getUserByAnyEmail(email, projection = {}) {
|
|
email = email.trim()
|
|
|
|
// $exists: true MUST be set to use the partial index
|
|
const query = { emails: { $exists: true }, 'emails.email': email }
|
|
const user = await db.users.findOne(query, { projection })
|
|
if (user) return user
|
|
|
|
// While multiple emails are being rolled out, check for the main email as
|
|
// well
|
|
return await getUserByMainEmail(email, projection)
|
|
}
|
|
|
|
async function getUsersByAnyConfirmedEmail(emails, projection = {}) {
|
|
const query = {
|
|
'emails.email': { $in: emails }, // use the index on emails.email
|
|
emails: {
|
|
$exists: true,
|
|
$elemMatch: {
|
|
email: { $in: emails },
|
|
confirmedAt: { $exists: true },
|
|
},
|
|
},
|
|
}
|
|
|
|
return await db.users.find(query, { projection }).toArray()
|
|
}
|
|
|
|
async function getUsersByV1Ids(v1Ids, projection = {}) {
|
|
const query = { 'overleaf.id': { $in: v1Ids } }
|
|
return await db.users.find(query, { projection }).toArray()
|
|
}
|
|
|
|
async function getUsersByHostname(hostname, projection) {
|
|
const reversedHostname = hostname.trim().split('').reverse().join('')
|
|
const query = {
|
|
emails: { $exists: true },
|
|
'emails.reversedHostname': reversedHostname,
|
|
}
|
|
return await db.users.find(query, { projection }).toArray()
|
|
}
|
|
|
|
async function getInstitutionUsersByHostname(hostname) {
|
|
const projection = {
|
|
_id: 1,
|
|
email: 1,
|
|
emails: 1,
|
|
samlIdentifiers: 1,
|
|
}
|
|
|
|
const users = await UserGetter.promises.getUsersByHostname(
|
|
hostname,
|
|
projection
|
|
)
|
|
users.forEach(user => {
|
|
user.emails = decorateFullEmails(
|
|
user.email,
|
|
user.emails,
|
|
[],
|
|
user.samlIdentifiers || []
|
|
)
|
|
})
|
|
return users
|
|
}
|
|
|
|
async function getUsers(query, projection) {
|
|
query = normalizeMultiQuery(query)
|
|
if (query?._id?.$in?.length === 0) return [] // shortcut for getUsers([])
|
|
return await db.users.find(query, { projection }).toArray()
|
|
}
|
|
|
|
// check for duplicate email address. This is also enforced at the DB level
|
|
async function ensureUniqueEmailAddress(newEmail) {
|
|
const user = await UserGetter.promises.getUserByAnyEmail(newEmail)
|
|
if (user) {
|
|
throw new Errors.EmailExistsError()
|
|
}
|
|
}
|
|
|
|
const UserGetter = {
|
|
getSsoUsersAtInstitution: callbackify(getSsoUsersAtInstitution),
|
|
getUser: callbackify(getUser),
|
|
getUserFeatures: callbackify(getUserFeatures),
|
|
getUserEmail: callbackify(getUserEmail),
|
|
getUserFullEmails: callbackify(getUserFullEmails),
|
|
getUserConfirmedEmails: callbackify(getUserConfirmedEmails),
|
|
getUserByMainEmail: callbackify(getUserByMainEmail),
|
|
getUserByAnyEmail: callbackify(getUserByAnyEmail),
|
|
getUsersByAnyConfirmedEmail: callbackify(getUsersByAnyConfirmedEmail),
|
|
getUsersByV1Ids: callbackify(getUsersByV1Ids),
|
|
getUsersByHostname: callbackify(getUsersByHostname),
|
|
getInstitutionUsersByHostname: callbackify(getInstitutionUsersByHostname),
|
|
getUsers: callbackify(getUsers),
|
|
// check for duplicate email address. This is also enforced at the DB level
|
|
ensureUniqueEmailAddress: callbackify(ensureUniqueEmailAddress),
|
|
getWritefullData: callbackify(getWritefullData),
|
|
}
|
|
|
|
const decorateFullEmails = (
|
|
defaultEmail,
|
|
emailsData,
|
|
affiliationsData,
|
|
samlIdentifiers
|
|
) => {
|
|
emailsData.forEach(function (emailData) {
|
|
emailData.default = emailData.email === defaultEmail
|
|
|
|
const affiliation = affiliationsData.find(
|
|
aff => aff.email === emailData.email
|
|
)
|
|
if (affiliation) {
|
|
const {
|
|
institution,
|
|
inferred,
|
|
role,
|
|
department,
|
|
licence,
|
|
cached_confirmed_at: cachedConfirmedAt,
|
|
cached_reconfirmed_at: cachedReconfirmedAt,
|
|
past_reconfirm_date: cachedPastReconfirmDate,
|
|
entitlement: cachedEntitlement,
|
|
portal,
|
|
group,
|
|
} = affiliation
|
|
const lastDayToReconfirm = _lastDayToReconfirm(emailData, institution)
|
|
let { last_day_to_reconfirm: cachedLastDayToReconfirm } = affiliation
|
|
if (institution.ssoEnabled && !emailData.samlProviderId) {
|
|
// only SSO linked emails are reconfirmed at SSO institutions
|
|
cachedLastDayToReconfirm = undefined
|
|
}
|
|
const pastReconfirmDate = _pastReconfirmDate(lastDayToReconfirm)
|
|
const inReconfirmNotificationPeriod = _emailInReconfirmNotificationPeriod(
|
|
cachedLastDayToReconfirm,
|
|
lastDayToReconfirm
|
|
)
|
|
emailData.affiliation = {
|
|
institution,
|
|
inferred,
|
|
inReconfirmNotificationPeriod,
|
|
lastDayToReconfirm,
|
|
cachedConfirmedAt,
|
|
cachedLastDayToReconfirm,
|
|
cachedReconfirmedAt,
|
|
cachedEntitlement,
|
|
cachedPastReconfirmDate,
|
|
pastReconfirmDate,
|
|
role,
|
|
department,
|
|
licence,
|
|
portal,
|
|
}
|
|
if (group) {
|
|
emailData.affiliation.group = group
|
|
}
|
|
}
|
|
|
|
if (emailData.samlProviderId) {
|
|
emailData.samlIdentifier = samlIdentifiers.find(
|
|
samlIdentifier => samlIdentifier.providerId === emailData.samlProviderId
|
|
)
|
|
}
|
|
|
|
emailData.emailHasInstitutionLicence =
|
|
InstitutionsHelper.emailHasLicence(emailData)
|
|
|
|
const lastConfirmedAtStr = emailData.reconfirmedAt || emailData.confirmedAt
|
|
emailData.lastConfirmedAt = lastConfirmedAtStr
|
|
? moment(lastConfirmedAtStr).toDate()
|
|
: null
|
|
})
|
|
|
|
return emailsData
|
|
}
|
|
|
|
UserGetter.promises = {
|
|
getSsoUsersAtInstitution,
|
|
getUser,
|
|
getUserFeatures,
|
|
getUserEmail,
|
|
getUserFullEmails,
|
|
getUserConfirmedEmails,
|
|
getUserByMainEmail,
|
|
getUserByAnyEmail,
|
|
getUsersByAnyConfirmedEmail,
|
|
getUsersByV1Ids,
|
|
getUsersByHostname,
|
|
getInstitutionUsersByHostname,
|
|
getUsers,
|
|
ensureUniqueEmailAddress,
|
|
getWritefullData,
|
|
}
|
|
|
|
export default UserGetter
|