overleaf/services/web/test/unit/src/Subscription/TeamInvitesHandlerTests.js

421 lines
12 KiB
JavaScript
Raw Normal View History

const SandboxedModule = require('sandboxed-module')
const sinon = require('sinon')
const { expect } = require('chai')
const modulePath =
'../../../../app/src/Features/Subscription/TeamInvitesHandler'
const { ObjectId } = require('mongodb')
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
}
this.subscription = {
id: '55153a8014829a865bbf700d',
_id: new ObjectId('55153a8014829a865bbf700d'),
admin_id: this.manager._id,
groupPlan: true,
member_ids: [],
teamInvites: [this.teamInvite],
save: sinon.stub().yields(null)
}
this.SubscriptionLocator = {
getUsersSubscription: sinon.stub(),
getSubscription: sinon.stub().yields(null, this.subscription)
}
this.UserGetter = {
getUser: sinon.stub().yields(),
getUserByAnyEmail: sinon.stub().yields()
}
this.SubscriptionUpdater = {
addUserToGroup: sinon.stub().yields()
}
this.LimitationsManager = {
teamHasReachedMemberLimit: sinon.stub().returns(false)
}
this.Subscription = {
findOne: sinon.stub().yields(),
updateOne: sinon.stub().yields()
}
this.EmailHandler = {
sendEmail: sinon.stub().yields(null)
}
this.newToken = 'bbbbbbbbb'
this.crypto = {
randomBytes: () => {
return { toString: sinon.stub().returns(this.newToken) }
}
}
this.UserGetter.getUser
.withArgs(this.manager._id)
.yields(null, this.manager)
this.UserGetter.getUserByAnyEmail
.withArgs(this.manager.email)
.yields(null, this.manager)
this.SubscriptionLocator.getUsersSubscription.yields(
null,
this.subscription
)
this.Subscription.findOne.yields(null, this.subscription)
this.TeamInvitesHandler = SandboxedModule.require(modulePath, {
globals: {
console: console
},
requires: {
mongodb: { ObjectId },
'logger-sharelatex': { log() {} },
crypto: this.crypto,
'settings-sharelatex': { siteUrl: 'http://example.com' },
'../../models/TeamInvite': { TeamInvite: (this.TeamInvite = {}) },
'../../models/Subscription': { Subscription: this.Subscription },
'../User/UserGetter': this.UserGetter,
'./SubscriptionLocator': this.SubscriptionLocator,
'./SubscriptionUpdater': this.SubscriptionUpdater,
'./LimitationsManager': this.LimitationsManager,
'../Email/EmailHandler': this.EmailHandler,
'../Errors/Errors': Errors
}
})
})
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().yields(null, 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.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.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)
}
)
})
})
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.getUserByAnyEmail
.withArgs(this.user.email)
.yields(null, this.user)
this.subscription.teamInvites.push({
email: 'john.snow@example.com',
token: 'dddddddd',
inviterName: 'Daenerys Targaryen (daenerys@example.com)'
})
})
it('adds the user to the team', function(done) {
this.TeamInvitesHandler.acceptInvite('dddddddd', this.user.id, () => {
this.SubscriptionUpdater.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()
})
})
})
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()
}
)
})
})
describe('createTeamInvitesForLegacyInvitedEmail', function(done) {
beforeEach(function() {
this.subscription.invited_emails = [
'eddard@example.com',
'robert@example.com'
]
this.TeamInvitesHandler.createInvite = sinon.stub().yields(null)
this.SubscriptionLocator.getGroupsWithEmailInvite = sinon
.stub()
.yields(null, [this.subscription])
})
it('sends an invitation email to addresses in the legacy invited_emails field', function(done) {
this.TeamInvitesHandler.createTeamInvitesForLegacyInvitedEmail(
'eddard@example.com',
(err, invite) => {
expect(err).not.to.exist
this.TeamInvitesHandler.createInvite
.calledWith(
this.subscription.admin_id,
this.subscription,
'eddard@example.com'
)
.should.eq(true)
this.TeamInvitesHandler.createInvite.callCount.should.eq(1)
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.getUserByAnyEmail
.withArgs(member.email)
.yields(null, 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()
}
)
})
})
})