Merge pull request #15582 from overleaf/msm-sso-linking-reminders

[web] SSO Linking redirects and interstitial info

GitOrigin-RevId: fdad0290ebf2b8b8a77f12b75736c030a4c7b642
This commit is contained in:
Alexandre Bourdin 2023-11-06 15:27:04 +01:00 committed by Copybot
parent 6852ad2be7
commit d45ca7d401
9 changed files with 48 additions and 3 deletions

View file

@ -2,6 +2,7 @@ const { promisify } = require('util')
const { Subscription } = require('../../models/Subscription') const { Subscription } = require('../../models/Subscription')
const { DeletedSubscription } = require('../../models/DeletedSubscription') const { DeletedSubscription } = require('../../models/DeletedSubscription')
const logger = require('@overleaf/logger') const logger = require('@overleaf/logger')
const { SSOConfig } = require('../../models/SSOConfig')
require('./GroupPlansData') // make sure dynamic group plans are loaded require('./GroupPlansData') // make sure dynamic group plans are loaded
const SubscriptionLocator = { const SubscriptionLocator = {
@ -124,6 +125,11 @@ const SubscriptionLocator = {
}, },
} }
async function hasSSOEnabled(subscription) {
const ssoConfig = await SSOConfig.findById(subscription.ssoConfig).exec()
return ssoConfig?.enabled
}
SubscriptionLocator.promises = { SubscriptionLocator.promises = {
getUsersSubscription: promisify(SubscriptionLocator.getUsersSubscription), getUsersSubscription: promisify(SubscriptionLocator.getUsersSubscription),
getUserIndividualSubscription: promisify( getUserIndividualSubscription: promisify(
@ -153,5 +159,6 @@ SubscriptionLocator.promises = {
hasRecurlyGroupSubscription: promisify( hasRecurlyGroupSubscription: promisify(
SubscriptionLocator.hasRecurlyGroupSubscription SubscriptionLocator.hasRecurlyGroupSubscription
), ),
hasSSOEnabled,
} }
module.exports = SubscriptionLocator module.exports = SubscriptionLocator

View file

@ -84,6 +84,10 @@ async function viewInvite(req, res, next) {
personalSubscription.recurlySubscription_id && personalSubscription.recurlySubscription_id &&
personalSubscription.recurlySubscription_id !== '' personalSubscription.recurlySubscription_id !== ''
const groupSSOActive = await SubscriptionLocator.promises.hasSSOEnabled(
subscription
)
if (subscription?.groupPolicy) { if (subscription?.groupPolicy) {
if (!subscription.populated('groupPolicy')) { if (!subscription.populated('groupPolicy')) {
await subscription.populate('groupPolicy') await subscription.populate('groupPolicy')
@ -128,6 +132,8 @@ async function viewInvite(req, res, next) {
expired: req.query.expired, expired: req.query.expired,
validationStatus: Object.fromEntries(validationStatus), validationStatus: Object.fromEntries(validationStatus),
currentManagedUserAdminEmail, currentManagedUserAdminEmail,
groupSSOActive,
subscriptionId: subscription._id.toString(),
}) })
} else { } else {
let currentManagedUserAdminEmail let currentManagedUserAdminEmail
@ -146,6 +152,8 @@ async function viewInvite(req, res, next) {
expired: req.query.expired, expired: req.query.expired,
userRestrictions: Array.from(req.userRestrictions || []), userRestrictions: Array.from(req.userRestrictions || []),
currentManagedUserAdminEmail, currentManagedUserAdminEmail,
groupSSOActive,
subscriptionId: subscription._id.toString(),
}) })
} }
} else { } else {
@ -167,8 +175,15 @@ async function acceptInvite(req, res, next) {
const { token } = req.params const { token } = req.params
const userId = SessionManager.getLoggedInUserId(req.session) const userId = SessionManager.getLoggedInUserId(req.session)
await TeamInvitesHandler.promises.acceptInvite(token, userId) const subscription = await TeamInvitesHandler.promises.acceptInvite(
res.sendStatus(204) token,
userId
)
const groupSSOActive = await SubscriptionLocator.promises.hasSSOEnabled(
subscription
)
res.status(204).json({ groupSSOActive })
} }
function revokeInvite(req, res, next) { function revokeInvite(req, res, next) {

View file

@ -95,6 +95,8 @@ async function acceptInvite(token, userId) {
await NotificationsBuilder.promises await NotificationsBuilder.promises
.groupInvitation(userId, subscription._id, false) .groupInvitation(userId, subscription._id, false)
.read() .read()
return subscription
} }
async function revokeInvite(teamManagerId, subscription, email) { async function revokeInvite(teamManagerId, subscription, email) {

View file

@ -10,6 +10,8 @@ block append meta
meta(name="ol-alreadyEnrolled" data-type="boolean" content=alreadyEnrolled) meta(name="ol-alreadyEnrolled" data-type="boolean" content=alreadyEnrolled)
meta(name="ol-validationStatus" data-type="json" content=validationStatus) meta(name="ol-validationStatus" data-type="json" content=validationStatus)
meta(name="ol-currentManagedUserAdminEmail" data-type="string" content=currentManagedUserAdminEmail) meta(name="ol-currentManagedUserAdminEmail" data-type="string" content=currentManagedUserAdminEmail)
meta(name="ol-groupSSOActive" data-type="boolean" content=groupSSOActive)
meta(name="ol-subscriptionId" data-type="string" content=subscriptionId)
block content block content
main.content.content-alt.team-invite#invite-managed-root main.content.content-alt.team-invite#invite-managed-root

View file

@ -3,6 +3,8 @@ extends ../../layout
block append meta block append meta
meta(name="ol-hasIndividualRecurlySubscription" data-type="boolean" content=hasIndividualRecurlySubscription) meta(name="ol-hasIndividualRecurlySubscription" data-type="boolean" content=hasIndividualRecurlySubscription)
meta(name="ol-inviteToken" content=inviteToken) meta(name="ol-inviteToken" content=inviteToken)
meta(name="ol-groupSSOActive" data-type="boolean" content=groupSSOActive)
meta(name="ol-subscriptionId" data-type="string" content=subscriptionId)
block content block content
main.content.content-alt.team-invite#main-content main.content.content-alt.team-invite#main-content
@ -42,4 +44,4 @@ block content
div(ng-show="view =='inviteAccepted'") div(ng-show="view =='inviteAccepted'")
p(ng-non-bindable) #{translate("joined_team", {inviterName: inviterName})} p(ng-non-bindable) #{translate("joined_team", {inviterName: inviterName})}
p p
a.btn.btn.btn-primary(href="/project") #{translate("done")} a.btn.btn.btn-primary(href=doneLink) #{translate("done")}

View file

@ -1421,6 +1421,7 @@
"you_need_to_configure_your_sso_settings": "", "you_need_to_configure_your_sso_settings": "",
"you_will_be_able_to_reassign_subscription": "", "you_will_be_able_to_reassign_subscription": "",
"youll_get_best_results_in_visual_but_can_be_used_in_source": "", "youll_get_best_results_in_visual_but_can_be_used_in_source": "",
"youll_no_longer_need_to_remember_credentials": "",
"your_affiliation_is_confirmed": "", "your_affiliation_is_confirmed": "",
"your_browser_does_not_support_this_feature": "", "your_browser_does_not_support_this_feature": "",
"your_compile_timed_out": "", "your_compile_timed_out": "",

View file

@ -24,6 +24,15 @@ export default App.controller('TeamInviteController', [
'ol-hasIndividualRecurlySubscription' 'ol-hasIndividualRecurlySubscription'
) )
const groupSSOActive = getMeta('ol-groupSSOActive', false)
if (groupSSOActive) {
const subscriptionId = getMeta('ol-subscriptionId')
$scope.doneLink = `/subscription/${subscriptionId}/sso_enrollment`
} else {
$scope.doneLink = '/project'
}
if (hideJoinSubscription) { if (hideJoinSubscription) {
$scope.view = 'restrictedByManagedGroup' $scope.view = 'restrictedByManagedGroup'
} else if (hasIndividualRecurlySubscription) { } else if (hasIndividualRecurlySubscription) {

View file

@ -2104,6 +2104,7 @@
"you_will_be_able_to_contact_us_any_time_to_share_your_feedback": "<0>You will be able to contact us</0> any time to share your feedback", "you_will_be_able_to_contact_us_any_time_to_share_your_feedback": "<0>You will be able to contact us</0> any time to share your feedback",
"you_will_be_able_to_reassign_subscription": "You will be able to reassign their subscription membership to another person in your organization", "you_will_be_able_to_reassign_subscription": "You will be able to reassign their subscription membership to another person in your organization",
"youll_get_best_results_in_visual_but_can_be_used_in_source": "Youll get the best results from using this tool in the <0>Visual Editor</0>, although you can still use it to insert tables in the <1>Code Editor</1>. Once youve selected the number of rows and columns you need, the table will appear in your document and you can double click in a cell to add contents to it.", "youll_get_best_results_in_visual_but_can_be_used_in_source": "Youll get the best results from using this tool in the <0>Visual Editor</0>, although you can still use it to insert tables in the <1>Code Editor</1>. Once youve selected the number of rows and columns you need, the table will appear in your document and you can double click in a cell to add contents to it.",
"youll_no_longer_need_to_remember_credentials": "Youll no longer need to remember a separate email address and password. Instead, you will use single-sign on to login to Overleaf. <0>Read more about SSO</0>.",
"your_account_is_managed_by_admin_cant_join_additional_group": "Your __appName__ account is managed by your current group admin (__admin__). This means you cant join additional group subscriptions. <0>Read more about Managed Users.</0>", "your_account_is_managed_by_admin_cant_join_additional_group": "Your __appName__ account is managed by your current group admin (__admin__). This means you cant join additional group subscriptions. <0>Read more about Managed Users.</0>",
"your_affiliation_is_confirmed": "Your <0>__institutionName__</0> affiliation is confirmed.", "your_affiliation_is_confirmed": "Your <0>__institutionName__</0> affiliation is confirmed.",
"your_browser_does_not_support_this_feature": "Sorry, your browser doesnt support this feature. Please update your browser to its latest version.", "your_browser_does_not_support_this_feature": "Sorry, your browser doesnt support this feature. Please update your browser to its latest version.",

View file

@ -15,6 +15,9 @@ describe('Subscription Locator Tests', function () {
findOne: sinon.stub().yields(), findOne: sinon.stub().yields(),
find: sinon.stub().yields(), find: sinon.stub().yields(),
} }
this.SSOConfig = {
findById: sinon.stub().yields(),
}
this.SubscriptionLocator = SandboxedModule.require(modulePath, { this.SubscriptionLocator = SandboxedModule.require(modulePath, {
requires: { requires: {
'./GroupPlansData': {}, './GroupPlansData': {},
@ -24,6 +27,9 @@ describe('Subscription Locator Tests', function () {
'../../models/DeletedSubscription': { '../../models/DeletedSubscription': {
DeletedSubscription: this.DeletedSubscription, DeletedSubscription: this.DeletedSubscription,
}, },
'../../models/SSOConfig': {
SSOConfig: this.SSOConfig,
},
}, },
}) })
}) })