2019-04-24 12:22:46 -04:00
|
|
|
Errors = require "../Errors/Errors"
|
|
|
|
User = require("../../models/User").User
|
2019-04-08 10:40:45 -04:00
|
|
|
UserStub = require("../../models/UserStub").UserStub
|
2019-04-24 12:22:46 -04:00
|
|
|
UserUpdater = require "./UserUpdater"
|
|
|
|
_ = require "lodash"
|
|
|
|
|
|
|
|
module.exports = ThirdPartyIdentityManager =
|
2019-05-13 11:35:10 -04:00
|
|
|
login: (providerId, externalUserId, externalData, callback) ->
|
2019-04-24 12:22:46 -04:00
|
|
|
return callback(new Error "invalid arguments") unless providerId? and externalUserId?
|
2019-05-13 11:35:10 -04:00
|
|
|
query = ThirdPartyIdentityManager._loginQuery providerId, externalUserId
|
2019-04-24 12:22:46 -04:00
|
|
|
User.findOne query, (err, user) ->
|
|
|
|
return callback err if err?
|
|
|
|
return callback(new Errors.ThirdPartyUserNotFoundError()) unless user
|
|
|
|
return callback(null, user) unless externalData
|
2019-05-13 11:35:10 -04:00
|
|
|
update = ThirdPartyIdentityManager._loginUpdate user, providerId, externalUserId, externalData
|
2019-04-24 12:22:46 -04:00
|
|
|
User.findOneAndUpdate query, update, {new: true}, callback
|
|
|
|
|
2019-04-08 10:40:45 -04:00
|
|
|
# attempt to login normally but check for user stub if user not found
|
|
|
|
loginUserStub: (providerId, externalUserId, externalData, callback) ->
|
|
|
|
ThirdPartyIdentityManager.login providerId, externalUserId, externalData, (err, user) ->
|
|
|
|
return callback null, user unless err?
|
|
|
|
return callback err unless err.name == "ThirdPartyUserNotFoundError"
|
2019-05-13 11:35:10 -04:00
|
|
|
query = ThirdPartyIdentityManager._loginQuery providerId, externalUserId
|
2019-04-08 10:40:45 -04:00
|
|
|
UserStub.findOne query, (err, userStub) ->
|
|
|
|
return callback err if err?
|
|
|
|
return callback(new Errors.ThirdPartyUserNotFoundError()) unless userStub
|
|
|
|
return callback(null, userStub) unless externalData
|
2019-05-13 11:35:10 -04:00
|
|
|
update = ThirdPartyIdentityManager._loginUpdate userStub, providerId, externalUserId, externalData
|
2019-04-08 10:40:45 -04:00
|
|
|
UserStub.findOneAndUpdate query, update, {new: true}, callback
|
|
|
|
|
2019-05-13 11:35:10 -04:00
|
|
|
_loginQuery: (providerId, externalUserId) ->
|
2019-04-08 10:40:45 -04:00
|
|
|
externalUserId = externalUserId.toString()
|
|
|
|
providerId = providerId.toString()
|
|
|
|
query =
|
|
|
|
"thirdPartyIdentifiers.externalUserId": externalUserId
|
|
|
|
"thirdPartyIdentifiers.providerId": providerId
|
|
|
|
return query
|
|
|
|
|
2019-05-13 11:35:10 -04:00
|
|
|
_loginUpdate: (user, providerId, externalUserId, externalData) ->
|
2019-04-08 10:40:45 -04:00
|
|
|
providerId = providerId.toString()
|
|
|
|
# get third party identifier object from array
|
|
|
|
thirdPartyIdentifier = user.thirdPartyIdentifiers.find (tpi) ->
|
|
|
|
tpi.externalUserId == externalUserId and tpi.providerId == providerId
|
|
|
|
# do recursive merge of new data over existing data
|
|
|
|
_.merge(thirdPartyIdentifier.externalData, externalData)
|
|
|
|
update = "thirdPartyIdentifiers.$": thirdPartyIdentifier
|
|
|
|
return update
|
|
|
|
|
2019-04-24 12:22:46 -04:00
|
|
|
# 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: (user_id, providerId, externalUserId, externalData, callback, retry) ->
|
|
|
|
query =
|
|
|
|
_id: user_id
|
|
|
|
"thirdPartyIdentifiers.providerId": $ne: providerId
|
|
|
|
update = $push: thirdPartyIdentifiers:
|
|
|
|
externalUserId: externalUserId
|
|
|
|
externalData: externalData
|
|
|
|
providerId: providerId
|
|
|
|
# add new tpi only if an entry for the provider does not exist
|
|
|
|
UserUpdater.updateUser query, update, (err, res) ->
|
|
|
|
return callback err if err?
|
|
|
|
return callback null, res if res.nModified == 1
|
|
|
|
# if already retried then throw error
|
|
|
|
return callback(new Error "update failed") if retry
|
|
|
|
# attempt to clear existing entry then retry
|
|
|
|
ThirdPartyIdentityManager.unlink user_id, providerId, (err) ->
|
|
|
|
return callback err if err?
|
|
|
|
ThirdPartyIdentityManager.link user_id, providerId, externalUserId, externalData, callback, true
|
|
|
|
|
|
|
|
unlink: (user_id, providerId, callback) ->
|
|
|
|
update = $pull: thirdPartyIdentifiers:
|
|
|
|
providerId: providerId
|
|
|
|
UserUpdater.updateUser user_id, update, callback
|