mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-05-31 21:01:33 +02:00
12 character strings are no longer valid see: https://github.com/mongodb/js-bson/releases/tag/v6.0.0 (Strings of length 12 can no longer make an ObjectId) GitOrigin-RevId: 2590c10915c50b951f991b2cf161d80235e445ca
602 lines
18 KiB
JavaScript
602 lines
18 KiB
JavaScript
const SandboxedModule = require('sandboxed-module')
|
|
const sinon = require('sinon')
|
|
const { expect } = require('chai')
|
|
const modulePath =
|
|
'../../../../app/src/Features/Subscription/TeamInvitesHandler'
|
|
|
|
const { ObjectId } = require('mongodb-legacy')
|
|
const Errors = require('../../../../app/src/Features/Errors/Errors')
|
|
|
|
describe('TeamInvitesHandler', function () {
|
|
beforeEach(function () {
|
|
this.manager = {
|
|
_id: '666666',
|
|
first_name: 'Daenerys',
|
|
last_name: 'Targaryen',
|
|
email: 'daenerys@example.com',
|
|
emails: [{ email: 'daenerys@example.com' }],
|
|
}
|
|
|
|
this.token = 'aaaaaaaaaaaaaaaaaaaaaa'
|
|
|
|
this.teamInvite = {
|
|
email: 'jorah@example.com',
|
|
token: this.token,
|
|
}
|
|
// ensure teamInvite can be converted from Document to Object
|
|
this.teamInvite.toObject = () => this.teamInvite
|
|
|
|
this.subscription = {
|
|
id: '55153a8014829a865bbf700d',
|
|
_id: new ObjectId('55153a8014829a865bbf700d'),
|
|
admin_id: this.manager._id,
|
|
groupPlan: true,
|
|
member_ids: [],
|
|
teamInvites: [this.teamInvite],
|
|
save: sinon.stub().resolves(),
|
|
}
|
|
|
|
this.SubscriptionLocator = {
|
|
promises: {
|
|
getUsersSubscription: sinon.stub(),
|
|
getSubscription: sinon.stub().resolves(this.subscription),
|
|
},
|
|
}
|
|
|
|
this.UserGetter = {
|
|
promises: {
|
|
getUser: sinon.stub().resolves(),
|
|
getUserByAnyEmail: sinon.stub().resolves(),
|
|
getUserByMainEmail: sinon.stub().resolves(),
|
|
},
|
|
}
|
|
|
|
this.SubscriptionUpdater = {
|
|
promises: {
|
|
addUserToGroup: sinon.stub().resolves(),
|
|
},
|
|
}
|
|
|
|
this.LimitationsManager = {
|
|
teamHasReachedMemberLimit: sinon.stub().returns(false),
|
|
}
|
|
|
|
this.Subscription = {
|
|
findOne: sinon.stub().resolves(),
|
|
updateOne: sinon.stub().resolves(),
|
|
}
|
|
|
|
this.SSOConfig = {
|
|
findById: sinon.stub().resolves(),
|
|
}
|
|
|
|
this.EmailHandler = {
|
|
promises: {
|
|
sendEmail: sinon.stub().resolves(null),
|
|
},
|
|
}
|
|
|
|
this.newToken = 'bbbbbbbbb'
|
|
|
|
this.crypto = {
|
|
randomBytes: () => {
|
|
return { toString: sinon.stub().returns(this.newToken) }
|
|
},
|
|
}
|
|
|
|
this.UserGetter.promises.getUser
|
|
.withArgs(this.manager._id)
|
|
.resolves(this.manager)
|
|
this.UserGetter.promises.getUserByAnyEmail
|
|
.withArgs(this.manager.email)
|
|
.resolves(this.manager)
|
|
this.UserGetter.promises.getUserByMainEmail
|
|
.withArgs(this.manager.email)
|
|
.resolves(this.manager)
|
|
|
|
this.SubscriptionLocator.promises.getUsersSubscription.resolves(
|
|
this.subscription
|
|
)
|
|
|
|
this.NotificationsBuilder = {
|
|
promises: {
|
|
groupInvitation: sinon.stub().returns({
|
|
create: sinon.stub().resolves(),
|
|
read: sinon.stub().resolves(),
|
|
}),
|
|
},
|
|
}
|
|
|
|
this.Subscription.findOne.resolves(this.subscription)
|
|
|
|
this.TeamInvitesHandler = SandboxedModule.require(modulePath, {
|
|
requires: {
|
|
'mongodb-legacy': { ObjectId },
|
|
crypto: this.crypto,
|
|
'@overleaf/settings': { siteUrl: 'http://example.com' },
|
|
'../../models/TeamInvite': { TeamInvite: (this.TeamInvite = {}) },
|
|
'../../models/Subscription': { Subscription: this.Subscription },
|
|
'../../models/SSOConfig': { SSOConfig: this.SSOConfig },
|
|
'../User/UserGetter': this.UserGetter,
|
|
'./SubscriptionLocator': this.SubscriptionLocator,
|
|
'./SubscriptionUpdater': this.SubscriptionUpdater,
|
|
'./LimitationsManager': this.LimitationsManager,
|
|
'../Email/EmailHandler': this.EmailHandler,
|
|
'../Notifications/NotificationsBuilder': this.NotificationsBuilder,
|
|
'../../infrastructure/Modules': (this.Modules = {
|
|
promises: { hooks: { fire: sinon.stub().resolves() } },
|
|
}),
|
|
},
|
|
})
|
|
})
|
|
|
|
describe('getInvite', function () {
|
|
it("returns the invite if there's one", function (done) {
|
|
this.TeamInvitesHandler.getInvite(
|
|
this.token,
|
|
(err, invite, subscription) => {
|
|
expect(err).to.eq(null)
|
|
expect(invite).to.deep.eq(this.teamInvite)
|
|
expect(subscription).to.deep.eq(this.subscription)
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
|
|
it("returns teamNotFound if there's none", function (done) {
|
|
this.Subscription.findOne = sinon.stub().resolves(null)
|
|
|
|
this.TeamInvitesHandler.getInvite(
|
|
this.token,
|
|
(err, invite, subscription) => {
|
|
expect(err).to.be.instanceof(Errors.NotFoundError)
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
})
|
|
|
|
describe('createInvite', function () {
|
|
it('adds the team invite to the subscription', function (done) {
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
'John.Snow@example.com',
|
|
(err, invite) => {
|
|
expect(err).to.eq(null)
|
|
expect(invite.token).to.eq(this.newToken)
|
|
expect(invite.email).to.eq('john.snow@example.com')
|
|
expect(invite.inviterName).to.eq(
|
|
'Daenerys Targaryen (daenerys@example.com)'
|
|
)
|
|
expect(invite.invite).to.be.true
|
|
expect(this.subscription.teamInvites).to.deep.include(invite)
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
|
|
it('sends an email', function (done) {
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
'John.Snow@example.com',
|
|
(err, invite) => {
|
|
this.EmailHandler.promises.sendEmail
|
|
.calledWith(
|
|
'verifyEmailToJoinTeam',
|
|
sinon.match({
|
|
to: 'john.snow@example.com',
|
|
inviter: this.manager,
|
|
acceptInviteUrl: `http://example.com/subscription/invites/${this.newToken}/`,
|
|
})
|
|
)
|
|
.should.equal(true)
|
|
done(err)
|
|
}
|
|
)
|
|
})
|
|
|
|
it('refreshes the existing invite if the email has already been invited', function (done) {
|
|
const originalInvite = Object.assign({}, this.teamInvite)
|
|
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
originalInvite.email,
|
|
(err, invite) => {
|
|
expect(err).to.eq(null)
|
|
expect(invite).to.exist
|
|
|
|
expect(this.subscription.teamInvites.length).to.eq(1)
|
|
expect(this.subscription.teamInvites).to.deep.include(invite)
|
|
|
|
expect(invite.email).to.eq(originalInvite.email)
|
|
|
|
this.subscription.save.calledOnce.should.eq(true)
|
|
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
|
|
it('removes any legacy invite from the subscription', function (done) {
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
'John.Snow@example.com',
|
|
(err, invite) => {
|
|
this.Subscription.updateOne
|
|
.calledWith(
|
|
{ _id: new ObjectId('55153a8014829a865bbf700d') },
|
|
{ $pull: { invited_emails: 'john.snow@example.com' } }
|
|
)
|
|
.should.eq(true)
|
|
done(err)
|
|
}
|
|
)
|
|
})
|
|
|
|
it('add user to subscription if inviting self', function (done) {
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
this.manager.email,
|
|
(err, invite) => {
|
|
sinon.assert.calledWith(
|
|
this.SubscriptionUpdater.promises.addUserToGroup,
|
|
this.subscription._id,
|
|
this.manager._id
|
|
)
|
|
sinon.assert.notCalled(this.subscription.save)
|
|
expect(invite.token).to.not.exist
|
|
expect(invite.email).to.eq(this.manager.email)
|
|
expect(invite.first_name).to.eq(this.manager.first_name)
|
|
expect(invite.last_name).to.eq(this.manager.last_name)
|
|
expect(invite.invite).to.be.false
|
|
done(err)
|
|
}
|
|
)
|
|
})
|
|
|
|
it('sends an SSO invite if SSO is enabled and inviting self', function (done) {
|
|
this.subscription.ssoConfig = new ObjectId('abc123abc123abc123abc123')
|
|
this.SSOConfig.findById
|
|
.withArgs(this.subscription.ssoConfig)
|
|
.resolves({ enabled: true })
|
|
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
this.manager.email,
|
|
(err, invite) => {
|
|
sinon.assert.calledWith(
|
|
this.Modules.promises.hooks.fire,
|
|
'sendGroupSSOReminder',
|
|
this.manager._id,
|
|
this.subscription._id
|
|
)
|
|
done(err)
|
|
}
|
|
)
|
|
})
|
|
|
|
it('does not send an SSO invite if SSO is disabled and inviting self', function (done) {
|
|
this.subscription.ssoConfig = new ObjectId('abc123abc123abc123abc123')
|
|
this.SSOConfig.findById
|
|
.withArgs(this.subscription.ssoConfig)
|
|
.resolves({ enabled: false })
|
|
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
this.manager.email,
|
|
(err, invite) => {
|
|
sinon.assert.notCalled(this.Modules.promises.hooks.fire)
|
|
done(err)
|
|
}
|
|
)
|
|
})
|
|
|
|
it('sends a notification if inviting registered user', function (done) {
|
|
const id = new ObjectId('6a6b3a8014829a865bbf700d')
|
|
const managedUsersEnabled = false
|
|
|
|
this.UserGetter.promises.getUserByMainEmail
|
|
.withArgs('john.snow@example.com')
|
|
.resolves({
|
|
_id: id,
|
|
})
|
|
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
'John.Snow@example.com',
|
|
(err, invite) => {
|
|
this.NotificationsBuilder.promises
|
|
.groupInvitation(
|
|
id.toString(),
|
|
this.subscription._id,
|
|
managedUsersEnabled
|
|
)
|
|
.create.calledWith(invite)
|
|
.should.eq(true)
|
|
done(err)
|
|
}
|
|
)
|
|
})
|
|
})
|
|
|
|
describe('importInvite', function () {
|
|
beforeEach(function () {
|
|
this.sentAt = new Date()
|
|
})
|
|
|
|
it('can imports an invite from v1', function () {
|
|
this.TeamInvitesHandler.importInvite(
|
|
this.subscription,
|
|
'A-Team',
|
|
'hannibal@a-team.org',
|
|
'secret',
|
|
this.sentAt,
|
|
error => {
|
|
expect(error).not.to.exist
|
|
|
|
this.subscription.save.calledOnce.should.eq(true)
|
|
|
|
const invite = this.subscription.teamInvites.find(
|
|
i => i.email === 'hannibal@a-team.org'
|
|
)
|
|
expect(invite.token).to.eq('secret')
|
|
expect(invite.sentAt).to.eq(this.sentAt)
|
|
}
|
|
)
|
|
})
|
|
})
|
|
|
|
describe('acceptInvite', function () {
|
|
beforeEach(function () {
|
|
this.user = {
|
|
id: '123456789',
|
|
first_name: 'Tyrion',
|
|
last_name: 'Lannister',
|
|
email: 'tyrion@example.com',
|
|
}
|
|
|
|
this.UserGetter.promises.getUserByAnyEmail
|
|
.withArgs(this.user.email)
|
|
.resolves(this.user)
|
|
|
|
this.subscription.teamInvites.push({
|
|
email: 'john.snow@example.com',
|
|
token: 'dddddddd',
|
|
inviterName: 'Daenerys Targaryen (daenerys@example.com)',
|
|
})
|
|
})
|
|
|
|
describe('with standard group', function () {
|
|
it('adds the user to the team', function (done) {
|
|
this.TeamInvitesHandler.acceptInvite('dddddddd', this.user.id, () => {
|
|
this.SubscriptionUpdater.promises.addUserToGroup
|
|
.calledWith(this.subscription._id, this.user.id)
|
|
.should.eq(true)
|
|
done()
|
|
})
|
|
})
|
|
|
|
it('removes the invite from the subscription', function (done) {
|
|
this.TeamInvitesHandler.acceptInvite('dddddddd', this.user.id, () => {
|
|
this.Subscription.updateOne
|
|
.calledWith(
|
|
{ _id: new ObjectId('55153a8014829a865bbf700d') },
|
|
{ $pull: { teamInvites: { email: 'john.snow@example.com' } } }
|
|
)
|
|
.should.eq(true)
|
|
done()
|
|
})
|
|
})
|
|
|
|
it('removes dashboard notification after they accepted group invitation', function (done) {
|
|
const managedUsersEnabled = false
|
|
|
|
this.TeamInvitesHandler.acceptInvite('dddddddd', this.user.id, () => {
|
|
sinon.assert.called(
|
|
this.NotificationsBuilder.promises.groupInvitation(
|
|
this.user.id,
|
|
this.subscription._id,
|
|
managedUsersEnabled
|
|
).read
|
|
)
|
|
done()
|
|
})
|
|
})
|
|
|
|
it('should not schedule an SSO invite reminder', function (done) {
|
|
this.TeamInvitesHandler.acceptInvite('dddddddd', this.user.id, () => {
|
|
sinon.assert.notCalled(this.Modules.promises.hooks.fire)
|
|
done()
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('with managed group', function () {
|
|
it('should enroll the group member', function (done) {
|
|
this.subscription.managedUsersEnabled = true
|
|
|
|
this.TeamInvitesHandler.acceptInvite('dddddddd', this.user.id, () => {
|
|
sinon.assert.calledWith(
|
|
this.Modules.promises.hooks.fire,
|
|
'enrollInManagedSubscription',
|
|
this.user.id,
|
|
this.subscription
|
|
)
|
|
done()
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('with group SSO enabled', function () {
|
|
it('should schedule an SSO invite reminder', function (done) {
|
|
this.subscription.ssoConfig = 'ssoconfig1'
|
|
this.SSOConfig.findById
|
|
.withArgs('ssoconfig1')
|
|
.resolves({ enabled: true })
|
|
|
|
this.TeamInvitesHandler.acceptInvite('dddddddd', this.user.id, () => {
|
|
sinon.assert.calledWith(
|
|
this.Modules.promises.hooks.fire,
|
|
'scheduleGroupSSOReminder',
|
|
this.user.id,
|
|
this.subscription._id
|
|
)
|
|
done()
|
|
})
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('revokeInvite', function () {
|
|
it('removes the team invite from the subscription', function (done) {
|
|
this.TeamInvitesHandler.revokeInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
'jorah@example.com',
|
|
() => {
|
|
this.Subscription.updateOne
|
|
.calledWith(
|
|
{ _id: new ObjectId('55153a8014829a865bbf700d') },
|
|
{ $pull: { teamInvites: { email: 'jorah@example.com' } } }
|
|
)
|
|
.should.eq(true)
|
|
|
|
this.Subscription.updateOne
|
|
.calledWith(
|
|
{ _id: new ObjectId('55153a8014829a865bbf700d') },
|
|
{ $pull: { invited_emails: 'jorah@example.com' } }
|
|
)
|
|
.should.eq(true)
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
|
|
it('removes dashboard notification for pending group invitation', function (done) {
|
|
const managedUsersEnabled = false
|
|
|
|
const pendingUser = {
|
|
id: '1a2b',
|
|
email: 'tyrion@example.com',
|
|
}
|
|
|
|
this.UserGetter.promises.getUserByAnyEmail
|
|
.withArgs(pendingUser.email)
|
|
.resolves(pendingUser)
|
|
|
|
this.TeamInvitesHandler.revokeInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
pendingUser.email,
|
|
() => {
|
|
sinon.assert.called(
|
|
this.NotificationsBuilder.promises.groupInvitation(
|
|
pendingUser.id,
|
|
this.subscription._id,
|
|
managedUsersEnabled
|
|
).read
|
|
)
|
|
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
})
|
|
|
|
describe('createTeamInvitesForLegacyInvitedEmail', function (done) {
|
|
beforeEach(function () {
|
|
this.subscription.invited_emails = [
|
|
'eddard@example.com',
|
|
'robert@example.com',
|
|
]
|
|
this.TeamInvitesHandler.createInvite = sinon.stub().resolves(null)
|
|
this.SubscriptionLocator.promises.getGroupsWithEmailInvite = sinon
|
|
.stub()
|
|
.resolves([this.subscription])
|
|
})
|
|
|
|
it('sends an invitation email to addresses in the legacy invited_emails field', function (done) {
|
|
this.TeamInvitesHandler.createTeamInvitesForLegacyInvitedEmail(
|
|
'eddard@example.com',
|
|
(err, invites) => {
|
|
expect(err).not.to.exist
|
|
expect(invites.length).to.eq(1)
|
|
|
|
const [invite] = invites
|
|
expect(invite.token).to.eq(this.newToken)
|
|
expect(invite.email).to.eq('eddard@example.com')
|
|
expect(invite.inviterName).to.eq(
|
|
'Daenerys Targaryen (daenerys@example.com)'
|
|
)
|
|
expect(invite.invite).to.be.true
|
|
expect(this.subscription.teamInvites).to.deep.include(invite)
|
|
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
})
|
|
|
|
describe('validation', function () {
|
|
it("doesn't create an invite if the team limit has been reached", function (done) {
|
|
this.LimitationsManager.teamHasReachedMemberLimit = sinon
|
|
.stub()
|
|
.returns(true)
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
'John.Snow@example.com',
|
|
(err, invite) => {
|
|
expect(err).to.deep.equal({ limitReached: true })
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
|
|
it("doesn't create an invite if the subscription is not in a group plan", function (done) {
|
|
this.subscription.groupPlan = false
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
'John.Snow@example.com',
|
|
(err, invite) => {
|
|
expect(err).to.deep.equal({ wrongPlan: true })
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
|
|
it("doesn't create an invite if the user is already part of the team", function (done) {
|
|
const member = {
|
|
id: '1a2b',
|
|
_id: '1a2b',
|
|
email: 'tyrion@example.com',
|
|
}
|
|
|
|
this.subscription.member_ids = [member.id]
|
|
this.UserGetter.promises.getUserByAnyEmail
|
|
.withArgs(member.email)
|
|
.resolves(member)
|
|
|
|
this.TeamInvitesHandler.createInvite(
|
|
this.manager._id,
|
|
this.subscription,
|
|
'tyrion@example.com',
|
|
(err, invite) => {
|
|
expect(err).to.deep.equal({ alreadyInTeam: true })
|
|
expect(invite).not.to.exist
|
|
done()
|
|
}
|
|
)
|
|
})
|
|
})
|
|
})
|