mirror of
https://github.com/overleaf/overleaf.git
synced 2024-11-21 20:47:08 -05:00
finish saml link after sso login
GitOrigin-RevId: 688ce78ddfb0cfd6a025985dc2dd6f62bce76523
This commit is contained in:
parent
9fa858bbd6
commit
8d44f44784
2 changed files with 182 additions and 168 deletions
|
@ -1,184 +1,196 @@
|
||||||
const EmailHandler = require('../../../../app/src/Features/Email/EmailHandler')
|
const APP_ROOT = '../../../../app/src'
|
||||||
|
const EmailHandler = require(`${APP_ROOT}/Features/Email/EmailHandler`)
|
||||||
const Errors = require('../Errors/Errors')
|
const Errors = require('../Errors/Errors')
|
||||||
const logger = require('logger-sharelatex')
|
|
||||||
const { User } = require('../../models/User')
|
|
||||||
const settings = require('settings-sharelatex')
|
|
||||||
const _ = require('lodash')
|
const _ = require('lodash')
|
||||||
|
const logger = require('logger-sharelatex')
|
||||||
|
const settings = require('settings-sharelatex')
|
||||||
|
const { User } = require(`${APP_ROOT}/models/User`)
|
||||||
|
const { promisifyAll } = require(`${APP_ROOT}/util/promises`)
|
||||||
|
|
||||||
const oauthProviders = settings.oauthProviders || {}
|
const oauthProviders = settings.oauthProviders || {}
|
||||||
|
|
||||||
const ThirdPartyIdentityManager = (module.exports = {
|
function getUser(providerId, externalUserId, callback) {
|
||||||
getUser(providerId, externalUserId, callback) {
|
if (providerId == null || externalUserId == null) {
|
||||||
if (providerId == null || externalUserId == null) {
|
return callback(new Error('invalid arguments'))
|
||||||
return callback(new Error('invalid arguments'))
|
}
|
||||||
|
const query = _getUserQuery(providerId, externalUserId)
|
||||||
|
User.findOne(query, function(err, user) {
|
||||||
|
if (err != null) {
|
||||||
|
return callback(err)
|
||||||
}
|
}
|
||||||
const query = ThirdPartyIdentityManager._getUserQuery(
|
if (!user) {
|
||||||
|
return callback(new Errors.ThirdPartyUserNotFoundError())
|
||||||
|
}
|
||||||
|
callback(null, user)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function login(providerId, externalUserId, externalData, callback) {
|
||||||
|
ThirdPartyIdentityManager.getUser(providerId, externalUserId, function(
|
||||||
|
err,
|
||||||
|
user
|
||||||
|
) {
|
||||||
|
if (err != null) {
|
||||||
|
return callback(err)
|
||||||
|
}
|
||||||
|
if (!externalData) {
|
||||||
|
return callback(null, user)
|
||||||
|
}
|
||||||
|
const query = _getUserQuery(providerId, externalUserId)
|
||||||
|
const update = _thirdPartyIdentifierUpdate(
|
||||||
|
user,
|
||||||
providerId,
|
providerId,
|
||||||
externalUserId
|
externalUserId,
|
||||||
|
externalData
|
||||||
)
|
)
|
||||||
User.findOne(query, function(err, user) {
|
User.findOneAndUpdate(query, update, { new: true }, callback)
|
||||||
if (err != null) {
|
})
|
||||||
return callback(err)
|
}
|
||||||
}
|
|
||||||
if (!user) {
|
|
||||||
return callback(new Errors.ThirdPartyUserNotFoundError())
|
|
||||||
}
|
|
||||||
callback(null, user)
|
|
||||||
})
|
|
||||||
},
|
|
||||||
|
|
||||||
login(providerId, externalUserId, externalData, callback) {
|
function link(
|
||||||
ThirdPartyIdentityManager.getUser(providerId, externalUserId, function(
|
userId,
|
||||||
err,
|
providerId,
|
||||||
user
|
externalUserId,
|
||||||
) {
|
externalData,
|
||||||
if (err != null) {
|
callback,
|
||||||
return callback(err)
|
retry
|
||||||
}
|
) {
|
||||||
if (!externalData) {
|
if (!oauthProviders[providerId]) {
|
||||||
return callback(null, user)
|
return callback(new Error('Not a valid provider'))
|
||||||
}
|
}
|
||||||
const query = ThirdPartyIdentityManager._getUserQuery(
|
const query = {
|
||||||
providerId,
|
_id: userId,
|
||||||
externalUserId
|
'thirdPartyIdentifiers.providerId': {
|
||||||
)
|
$ne: providerId
|
||||||
const update = ThirdPartyIdentityManager._thirdPartyIdentifierUpdate(
|
}
|
||||||
user,
|
}
|
||||||
providerId,
|
const update = {
|
||||||
|
$push: {
|
||||||
|
thirdPartyIdentifiers: {
|
||||||
externalUserId,
|
externalUserId,
|
||||||
externalData
|
externalData,
|
||||||
)
|
providerId
|
||||||
User.findOneAndUpdate(query, update, { new: true }, callback)
|
|
||||||
})
|
|
||||||
},
|
|
||||||
|
|
||||||
_getUserQuery(providerId, externalUserId) {
|
|
||||||
externalUserId = externalUserId.toString()
|
|
||||||
providerId = providerId.toString()
|
|
||||||
const query = {
|
|
||||||
'thirdPartyIdentifiers.externalUserId': externalUserId,
|
|
||||||
'thirdPartyIdentifiers.providerId': providerId
|
|
||||||
}
|
|
||||||
return query
|
|
||||||
},
|
|
||||||
|
|
||||||
_thirdPartyIdentifierUpdate(user, providerId, externalUserId, externalData) {
|
|
||||||
providerId = providerId.toString()
|
|
||||||
// get third party identifier object from array
|
|
||||||
const thirdPartyIdentifier = user.thirdPartyIdentifiers.find(
|
|
||||||
tpi =>
|
|
||||||
tpi.externalUserId === externalUserId && tpi.providerId === providerId
|
|
||||||
)
|
|
||||||
// do recursive merge of new data over existing data
|
|
||||||
_.merge(thirdPartyIdentifier.externalData, externalData)
|
|
||||||
const update = { 'thirdPartyIdentifiers.$': thirdPartyIdentifier }
|
|
||||||
return update
|
|
||||||
},
|
|
||||||
|
|
||||||
// register: () ->
|
|
||||||
// this should be implemented once we move to having v2 as the master
|
|
||||||
// but for now we need to register with v1 then call link once that
|
|
||||||
// is complete
|
|
||||||
|
|
||||||
link(userId, providerId, externalUserId, externalData, callback, retry) {
|
|
||||||
if (!oauthProviders[providerId]) {
|
|
||||||
return callback(new Error('Not a valid provider'))
|
|
||||||
}
|
|
||||||
const query = {
|
|
||||||
_id: userId,
|
|
||||||
'thirdPartyIdentifiers.providerId': {
|
|
||||||
$ne: providerId
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const update = {
|
}
|
||||||
$push: {
|
// add new tpi only if an entry for the provider does not exist
|
||||||
thirdPartyIdentifiers: {
|
// projection includes thirdPartyIdentifiers for tests
|
||||||
|
User.findOneAndUpdate(query, update, { new: 1 }, (err, res) => {
|
||||||
|
if (err && err.code === 11000) {
|
||||||
|
callback(new Errors.ThirdPartyIdentityExistsError())
|
||||||
|
} else if (err != null) {
|
||||||
|
callback(err)
|
||||||
|
} else if (res) {
|
||||||
|
const emailOptions = {
|
||||||
|
to: res.email,
|
||||||
|
provider: oauthProviders[providerId].name
|
||||||
|
}
|
||||||
|
EmailHandler.sendEmail(
|
||||||
|
'emailThirdPartyIdentifierLinked',
|
||||||
|
emailOptions,
|
||||||
|
error => {
|
||||||
|
if (error != null) {
|
||||||
|
logger.warn(error)
|
||||||
|
}
|
||||||
|
return callback(null, res)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
} else if (retry) {
|
||||||
|
// if already retried then throw error
|
||||||
|
callback(new Error('update failed'))
|
||||||
|
} else {
|
||||||
|
// attempt to clear existing entry then retry
|
||||||
|
ThirdPartyIdentityManager.unlink(userId, providerId, function(err) {
|
||||||
|
if (err != null) {
|
||||||
|
return callback(err)
|
||||||
|
}
|
||||||
|
ThirdPartyIdentityManager.link(
|
||||||
|
userId,
|
||||||
|
providerId,
|
||||||
externalUserId,
|
externalUserId,
|
||||||
externalData,
|
externalData,
|
||||||
providerId
|
callback,
|
||||||
}
|
true
|
||||||
}
|
|
||||||
}
|
|
||||||
// add new tpi only if an entry for the provider does not exist
|
|
||||||
// projection includes thirdPartyIdentifiers for tests
|
|
||||||
User.findOneAndUpdate(query, update, { new: 1 }, (err, res) => {
|
|
||||||
if (err && err.code === 11000) {
|
|
||||||
callback(new Errors.ThirdPartyIdentityExistsError())
|
|
||||||
} else if (err != null) {
|
|
||||||
callback(err)
|
|
||||||
} else if (res) {
|
|
||||||
const emailOptions = {
|
|
||||||
to: res.email,
|
|
||||||
provider: oauthProviders[providerId].name
|
|
||||||
}
|
|
||||||
EmailHandler.sendEmail(
|
|
||||||
'emailThirdPartyIdentifierLinked',
|
|
||||||
emailOptions,
|
|
||||||
error => {
|
|
||||||
if (error != null) {
|
|
||||||
logger.warn(error)
|
|
||||||
}
|
|
||||||
return callback(null, res)
|
|
||||||
}
|
|
||||||
)
|
)
|
||||||
} else if (retry) {
|
})
|
||||||
// if already retried then throw error
|
}
|
||||||
callback(new Error('update failed'))
|
})
|
||||||
} else {
|
}
|
||||||
// attempt to clear existing entry then retry
|
|
||||||
ThirdPartyIdentityManager.unlink(userId, providerId, function(err) {
|
|
||||||
if (err != null) {
|
|
||||||
return callback(err)
|
|
||||||
}
|
|
||||||
ThirdPartyIdentityManager.link(
|
|
||||||
userId,
|
|
||||||
providerId,
|
|
||||||
externalUserId,
|
|
||||||
externalData,
|
|
||||||
callback,
|
|
||||||
true
|
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
})
|
|
||||||
},
|
|
||||||
|
|
||||||
unlink(userId, providerId, callback) {
|
function unlink(userId, providerId, callback) {
|
||||||
if (!oauthProviders[providerId]) {
|
if (!oauthProviders[providerId]) {
|
||||||
return callback(new Error('Not a valid provider'))
|
return callback(new Error('Not a valid provider'))
|
||||||
}
|
|
||||||
const query = {
|
|
||||||
_id: userId
|
|
||||||
}
|
|
||||||
const update = {
|
|
||||||
$pull: {
|
|
||||||
thirdPartyIdentifiers: {
|
|
||||||
providerId
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// projection includes thirdPartyIdentifiers for tests
|
|
||||||
User.findOneAndUpdate(query, update, { new: 1 }, (err, res) => {
|
|
||||||
if (err != null) {
|
|
||||||
callback(err)
|
|
||||||
} else if (!res) {
|
|
||||||
callback(new Error('update failed'))
|
|
||||||
} else {
|
|
||||||
const emailOptions = {
|
|
||||||
to: res.email,
|
|
||||||
provider: oauthProviders[providerId].name
|
|
||||||
}
|
|
||||||
EmailHandler.sendEmail(
|
|
||||||
'emailThirdPartyIdentifierUnlinked',
|
|
||||||
emailOptions,
|
|
||||||
error => {
|
|
||||||
if (error != null) {
|
|
||||||
logger.warn(error)
|
|
||||||
}
|
|
||||||
return callback(null, res)
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
})
|
const query = {
|
||||||
|
_id: userId
|
||||||
|
}
|
||||||
|
const update = {
|
||||||
|
$pull: {
|
||||||
|
thirdPartyIdentifiers: {
|
||||||
|
providerId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// projection includes thirdPartyIdentifiers for tests
|
||||||
|
User.findOneAndUpdate(query, update, { new: 1 }, (err, res) => {
|
||||||
|
if (err != null) {
|
||||||
|
callback(err)
|
||||||
|
} else if (!res) {
|
||||||
|
callback(new Error('update failed'))
|
||||||
|
} else {
|
||||||
|
const emailOptions = {
|
||||||
|
to: res.email,
|
||||||
|
provider: oauthProviders[providerId].name
|
||||||
|
}
|
||||||
|
EmailHandler.sendEmail(
|
||||||
|
'emailThirdPartyIdentifierUnlinked',
|
||||||
|
emailOptions,
|
||||||
|
error => {
|
||||||
|
if (error != null) {
|
||||||
|
logger.warn(error)
|
||||||
|
}
|
||||||
|
return callback(null, res)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function _getUserQuery(providerId, externalUserId) {
|
||||||
|
externalUserId = externalUserId.toString()
|
||||||
|
providerId = providerId.toString()
|
||||||
|
const query = {
|
||||||
|
'thirdPartyIdentifiers.externalUserId': externalUserId,
|
||||||
|
'thirdPartyIdentifiers.providerId': providerId
|
||||||
|
}
|
||||||
|
return query
|
||||||
|
}
|
||||||
|
|
||||||
|
function _thirdPartyIdentifierUpdate(
|
||||||
|
user,
|
||||||
|
providerId,
|
||||||
|
externalUserId,
|
||||||
|
externalData
|
||||||
|
) {
|
||||||
|
providerId = providerId.toString()
|
||||||
|
// get third party identifier object from array
|
||||||
|
const thirdPartyIdentifier = user.thirdPartyIdentifiers.find(
|
||||||
|
tpi =>
|
||||||
|
tpi.externalUserId === externalUserId && tpi.providerId === providerId
|
||||||
|
)
|
||||||
|
// do recursive merge of new data over existing data
|
||||||
|
_.merge(thirdPartyIdentifier.externalData, externalData)
|
||||||
|
const update = { 'thirdPartyIdentifiers.$': thirdPartyIdentifier }
|
||||||
|
return update
|
||||||
|
}
|
||||||
|
|
||||||
|
const ThirdPartyIdentityManager = {
|
||||||
|
getUser,
|
||||||
|
login,
|
||||||
|
link,
|
||||||
|
unlink
|
||||||
|
}
|
||||||
|
|
||||||
|
ThirdPartyIdentityManager.promises = promisifyAll(ThirdPartyIdentityManager)
|
||||||
|
|
||||||
|
module.exports = ThirdPartyIdentityManager
|
||||||
|
|
|
@ -41,13 +41,15 @@ module.exports = class UserHelper {
|
||||||
this.user = null
|
this.user = null
|
||||||
// cookie jar
|
// cookie jar
|
||||||
this.jar = request.jar()
|
this.jar = request.jar()
|
||||||
|
// create new request instance
|
||||||
|
this.request = request.defaults({})
|
||||||
// initialize request instance with default options
|
// initialize request instance with default options
|
||||||
this.setRequestDefaults()
|
this.setRequestDefaults()
|
||||||
}
|
}
|
||||||
|
|
||||||
setRequestDefaults(defaults = {}) {
|
setRequestDefaults(defaults = {}) {
|
||||||
// request-promise instance for making requests
|
// request-promise instance for making requests
|
||||||
this.request = request.defaults({
|
this.request = this.request.defaults({
|
||||||
baseUrl: UserHelper.baseUrl(),
|
baseUrl: UserHelper.baseUrl(),
|
||||||
followRedirect: false,
|
followRedirect: false,
|
||||||
jar: this.jar,
|
jar: this.jar,
|
||||||
|
|
Loading…
Reference in a new issue