2019-05-29 05:21:06 -04:00
|
|
|
/* eslint-disable
|
|
|
|
handle-callback-err,
|
|
|
|
max-len,
|
|
|
|
no-return-assign,
|
|
|
|
no-unused-vars,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS102: Remove unnecessary code created because of implicit returns
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
|
|
|
|
*/
|
|
|
|
const should = require('chai').should()
|
|
|
|
const SandboxedModule = require('sandboxed-module')
|
|
|
|
const assert = require('assert')
|
|
|
|
const path = require('path')
|
|
|
|
const modulePath = path.join(
|
|
|
|
__dirname,
|
|
|
|
'../../../../app/src/Features/User/UserRegistrationHandler'
|
|
|
|
)
|
|
|
|
const sinon = require('sinon')
|
|
|
|
const { expect } = require('chai')
|
|
|
|
const EmailHelper = require('../../../../app/src/Features/Helpers/EmailHelper')
|
|
|
|
|
|
|
|
describe('UserRegistrationHandler', function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.user = { _id: (this.user_id = '31j2lk21kjl') }
|
|
|
|
this.User = { update: sinon.stub().callsArgWith(2) }
|
|
|
|
this.UserGetter = { getUserByAnyEmail: sinon.stub() }
|
|
|
|
this.UserCreator = {
|
2020-09-01 08:37:09 -04:00
|
|
|
createNewUser: sinon.stub().callsArgWith(2, null, this.user)
|
2019-05-29 05:21:06 -04:00
|
|
|
}
|
|
|
|
this.AuthenticationManager = {
|
|
|
|
validateEmail: sinon.stub().returns(null),
|
|
|
|
validatePassword: sinon.stub().returns(null),
|
|
|
|
setUserPassword: sinon.stub().callsArgWith(2)
|
|
|
|
}
|
|
|
|
this.NewsLetterManager = { subscribe: sinon.stub().callsArgWith(1) }
|
|
|
|
this.EmailHandler = { sendEmail: sinon.stub().callsArgWith(2) }
|
|
|
|
this.OneTimeTokenHandler = { getNewToken: sinon.stub() }
|
|
|
|
this.handler = SandboxedModule.require(modulePath, {
|
2019-07-15 06:33:47 -04:00
|
|
|
globals: {
|
|
|
|
console: console
|
|
|
|
},
|
2019-05-29 05:21:06 -04:00
|
|
|
requires: {
|
|
|
|
'../../models/User': { User: this.User },
|
|
|
|
'./UserGetter': this.UserGetter,
|
|
|
|
'./UserCreator': this.UserCreator,
|
|
|
|
'../Authentication/AuthenticationManager': this.AuthenticationManager,
|
|
|
|
'../Newsletter/NewsletterManager': this.NewsLetterManager,
|
|
|
|
'logger-sharelatex': (this.logger = { log: sinon.stub() }),
|
|
|
|
crypto: (this.crypto = {}),
|
|
|
|
'../Email/EmailHandler': this.EmailHandler,
|
|
|
|
'../Security/OneTimeTokenHandler': this.OneTimeTokenHandler,
|
|
|
|
'../Analytics/AnalyticsManager': (this.AnalyticsManager = {
|
|
|
|
recordEvent: sinon.stub()
|
|
|
|
}),
|
|
|
|
'settings-sharelatex': (this.settings = {
|
|
|
|
siteUrl: 'http://sl.example.com'
|
|
|
|
}),
|
|
|
|
'../Helpers/EmailHelper': EmailHelper
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return (this.passingRequest = {
|
|
|
|
email: 'something@email.com',
|
|
|
|
password: '123'
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('validate Register Request', function() {
|
|
|
|
it('allows passing validation through', function() {
|
|
|
|
const result = this.handler._registrationRequestIsValid(
|
|
|
|
this.passingRequest
|
|
|
|
)
|
|
|
|
return result.should.equal(true)
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('failing email validation', function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
return this.AuthenticationManager.validateEmail.returns({
|
|
|
|
message: 'email not set'
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
it('does not allow through', function() {
|
2019-05-29 05:21:06 -04:00
|
|
|
const result = this.handler._registrationRequestIsValid(
|
|
|
|
this.passingRequest
|
|
|
|
)
|
|
|
|
return result.should.equal(false)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
describe('failing password validation', function() {
|
2019-05-29 05:21:06 -04:00
|
|
|
beforeEach(function() {
|
|
|
|
return this.AuthenticationManager.validatePassword.returns({
|
|
|
|
message: 'password is too short'
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
it('does not allow through', function() {
|
2019-05-29 05:21:06 -04:00
|
|
|
const result = this.handler._registrationRequestIsValid(
|
|
|
|
this.passingRequest
|
|
|
|
)
|
|
|
|
return result.should.equal(false)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('registerNewUser', function() {
|
|
|
|
describe('holdingAccount', function(done) {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.user.holdingAccount = true
|
|
|
|
this.handler._registrationRequestIsValid = sinon.stub().returns(true)
|
|
|
|
return this.UserGetter.getUserByAnyEmail.callsArgWith(
|
|
|
|
1,
|
|
|
|
null,
|
|
|
|
this.user
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should not create a new user if there is a holding account there', function(done) {
|
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
this.UserCreator.createNewUser.called.should.equal(false)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
it('should set holding account to false', function(done) {
|
2019-05-29 05:21:06 -04:00
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
const update = this.User.update.args[0]
|
|
|
|
assert.deepEqual(update[0], { _id: this.user._id })
|
|
|
|
assert.deepEqual(update[1], { $set: { holdingAccount: false } })
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('invalidRequest', function() {
|
|
|
|
it('should not create a new user if the the request is not valid', function(done) {
|
|
|
|
this.handler._registrationRequestIsValid = sinon.stub().returns(false)
|
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
expect(err).to.exist
|
|
|
|
this.UserCreator.createNewUser.called.should.equal(false)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
it('should return email registered in the error if there is a non holdingAccount there', function(done) {
|
2019-05-29 05:21:06 -04:00
|
|
|
this.UserGetter.getUserByAnyEmail.callsArgWith(
|
|
|
|
1,
|
|
|
|
null,
|
|
|
|
(this.user = { holdingAccount: false })
|
|
|
|
)
|
|
|
|
return this.handler.registerNewUser(
|
|
|
|
this.passingRequest,
|
|
|
|
(err, user) => {
|
2019-11-18 09:37:05 -05:00
|
|
|
expect(err).to.be.instanceOf(Error)
|
|
|
|
expect(err).to.have.property('message', 'EmailAlreadyRegistered')
|
2019-05-29 05:21:06 -04:00
|
|
|
user.should.deep.equal(this.user)
|
|
|
|
return done()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('validRequest', function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.handler._registrationRequestIsValid = sinon.stub().returns(true)
|
|
|
|
return this.UserGetter.getUserByAnyEmail.callsArgWith(1)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should create a new user', function(done) {
|
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
this.UserCreator.createNewUser
|
|
|
|
.calledWith({
|
|
|
|
email: this.passingRequest.email,
|
|
|
|
holdingAccount: false,
|
|
|
|
first_name: this.passingRequest.first_name,
|
|
|
|
last_name: this.passingRequest.last_name
|
|
|
|
})
|
|
|
|
.should.equal(true)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('lower case email', function(done) {
|
|
|
|
this.passingRequest.email = 'soMe@eMail.cOm'
|
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
this.UserCreator.createNewUser.args[0][0].email.should.equal(
|
|
|
|
'some@email.com'
|
|
|
|
)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('trim white space from email', function(done) {
|
|
|
|
this.passingRequest.email = ' some@email.com '
|
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
this.UserCreator.createNewUser.args[0][0].email.should.equal(
|
|
|
|
'some@email.com'
|
|
|
|
)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should set the password', function(done) {
|
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
this.AuthenticationManager.setUserPassword
|
|
|
|
.calledWith(this.user._id, this.passingRequest.password)
|
|
|
|
.should.equal(true)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should add the user to the newsletter if accepted terms', function(done) {
|
|
|
|
this.passingRequest.subscribeToNewsletter = 'true'
|
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
this.NewsLetterManager.subscribe
|
|
|
|
.calledWith(this.user)
|
|
|
|
.should.equal(true)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should not add the user to the newsletter if not accepted terms', function(done) {
|
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
this.NewsLetterManager.subscribe
|
|
|
|
.calledWith(this.user)
|
|
|
|
.should.equal(false)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
it('should track the registration event', function(done) {
|
2019-05-29 05:21:06 -04:00
|
|
|
return this.handler.registerNewUser(this.passingRequest, err => {
|
|
|
|
this.AnalyticsManager.recordEvent
|
|
|
|
.calledWith(this.user._id, 'user-registered')
|
|
|
|
.should.equal(true)
|
|
|
|
return done()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-08-07 10:04:04 -04:00
|
|
|
it('should call the ReferalAllocator', function(done) {
|
|
|
|
return done()
|
|
|
|
})
|
2019-05-29 05:21:06 -04:00
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
describe('registerNewUserAndSendActivationEmail', function() {
|
2019-05-29 05:21:06 -04:00
|
|
|
beforeEach(function() {
|
|
|
|
this.email = 'email@example.com'
|
|
|
|
this.crypto.randomBytes = sinon.stub().returns({
|
|
|
|
toString: () => {
|
|
|
|
return (this.password = 'mock-password')
|
|
|
|
}
|
|
|
|
})
|
|
|
|
this.OneTimeTokenHandler.getNewToken.yields(
|
|
|
|
null,
|
|
|
|
(this.token = 'mock-token')
|
|
|
|
)
|
|
|
|
this.handler.registerNewUser = sinon.stub()
|
|
|
|
return (this.callback = sinon.stub())
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('with a new user', function() {
|
|
|
|
beforeEach(function() {
|
|
|
|
this.handler.registerNewUser.callsArgWith(1, null, this.user)
|
|
|
|
return this.handler.registerNewUserAndSendActivationEmail(
|
|
|
|
this.email,
|
|
|
|
this.callback
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should ask the UserRegistrationHandler to register user', function() {
|
|
|
|
return this.handler.registerNewUser
|
|
|
|
.calledWith({
|
|
|
|
email: this.email,
|
|
|
|
password: this.password
|
|
|
|
})
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should generate a new password reset token', function() {
|
2019-09-04 06:57:48 -04:00
|
|
|
const data = { user_id: this.user._id.toString(), email: this.email }
|
2019-05-29 05:21:06 -04:00
|
|
|
return this.OneTimeTokenHandler.getNewToken
|
2019-09-04 06:57:48 -04:00
|
|
|
.calledWith('password', data, { expiresIn: 7 * 24 * 60 * 60 })
|
2019-05-29 05:21:06 -04:00
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should send a registered email', function() {
|
|
|
|
return this.EmailHandler.sendEmail
|
|
|
|
.calledWith('registered', {
|
|
|
|
to: this.user.email,
|
|
|
|
setNewPasswordUrl: `${this.settings.siteUrl}/user/activate?token=${
|
|
|
|
this.token
|
|
|
|
}&user_id=${this.user_id}`
|
|
|
|
})
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
it('should return the user', function() {
|
2019-05-29 05:21:06 -04:00
|
|
|
return this.callback
|
|
|
|
.calledWith(
|
|
|
|
null,
|
|
|
|
this.user,
|
|
|
|
`${this.settings.siteUrl}/user/activate?token=${
|
|
|
|
this.token
|
|
|
|
}&user_id=${this.user_id}`
|
|
|
|
)
|
|
|
|
.should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
describe('with a user that already exists', function() {
|
2019-05-29 05:21:06 -04:00
|
|
|
beforeEach(function() {
|
|
|
|
this.handler.registerNewUser.callsArgWith(
|
|
|
|
1,
|
|
|
|
new Error('EmailAlreadyRegistered'),
|
|
|
|
this.user
|
|
|
|
)
|
|
|
|
return this.handler.registerNewUserAndSendActivationEmail(
|
|
|
|
this.email,
|
|
|
|
this.callback
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2019-06-21 09:46:09 -04:00
|
|
|
it('should still generate a new password token and email', function() {
|
2019-05-29 05:21:06 -04:00
|
|
|
this.OneTimeTokenHandler.getNewToken.called.should.equal(true)
|
|
|
|
return this.EmailHandler.sendEmail.called.should.equal(true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|