Skip to content

Commit

Permalink
Merge branch 'staging' into development
Browse files Browse the repository at this point in the history
  • Loading branch information
fnocetti committed May 23, 2024
2 parents 7c77bd0 + 8b5c5f7 commit 21c41c2
Show file tree
Hide file tree
Showing 5 changed files with 18 additions and 12 deletions.
3 changes: 3 additions & 0 deletions app/api/users/generateUnlockCode.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import crypto from 'crypto';

export const generateUnlockCode = () => crypto.randomBytes(32).toString('hex');
3 changes: 3 additions & 0 deletions app/api/users/passwordRecoveriesModel.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,12 @@ import mongoose from 'mongoose';

import { instanceModel } from 'api/odm';

const ONE_DAY = 60 * 60 * 24;

const schema = new mongoose.Schema({
key: String,
user: { type: mongoose.Schema.Types.ObjectId, ref: 'users' },
expiresAt: { type: Date, expires: ONE_DAY, default: Date.now() },
});

export default instanceModel('passwordrecoveries', schema);
11 changes: 6 additions & 5 deletions app/api/users/specs/users.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@
/* eslint-disable max-statements */

import { createError } from 'api/utils';
import SHA256 from 'crypto-js/sha256';
import crypto from 'crypto';
import mailer from 'api/utils/mailer';
import db from 'api/utils/testing_db';
Expand All @@ -25,6 +24,7 @@ import fixtures, {
import users from '../users.js';
import passwordRecoveriesModel from '../passwordRecoveriesModel';
import usersModel from '../usersModel';
import * as unlockCode from '../generateUnlockCode';

describe('Users', () => {
beforeEach(async () => {
Expand Down Expand Up @@ -510,10 +510,11 @@ describe('Users', () => {
jest.restoreAllMocks();
jest.spyOn(mailer, 'send').mockImplementation(async () => Promise.resolve('OK'));
jest.spyOn(Date, 'now').mockReturnValue(1000);
jest.spyOn(unlockCode, 'generateUnlockCode').mockReturnValue('ABCDEF1234');
});

it('should find the matching email create a recover password doc in the database and send an email', async () => {
const key = SHA256(`[email protected]${1000}`).toString();
const key = unlockCode.generateUnlockCode();
const settings = await settingsModel.get();
const response = await users.recoverPassword('[email protected]', 'domain');
expect(response).toBe('OK');
Expand All @@ -524,13 +525,13 @@ describe('Users', () => {
from: emailSender,
to: '[email protected]',
subject: 'Password set',
text: `To set your password click on the following link:\ndomain/setpassword/${key}`,
text: `To set your password click on the following link:\ndomain/setpassword/${key}\nThis link will be valid for 24 hours.`,
};
expect(mailer.send).toHaveBeenCalledWith(expectedMailOptions);
});

it('should personalize the mail if recover password process is part of a newly created user', async () => {
const key = SHA256(`[email protected]${1000}`).toString();
const key = unlockCode.generateUnlockCode();
const settings = await settingsModel.get();

const newUser = await users.newUser(
Expand Down Expand Up @@ -590,7 +591,7 @@ describe('Users', () => {
describe('when the user does not exist with that email', () => {
it('should not create the entry in the database, should not send a mail, and return an error.', async () => {
jest.spyOn(Date, 'now').mockReturnValue(1000);
const key = SHA256(`[email protected]${1000}`).toString();
const key = unlockCode.generateUnlockCode();
let response;
try {
response = await users.recoverPassword('[email protected]');
Expand Down
11 changes: 5 additions & 6 deletions app/api/users/users.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
import SHA256 from 'crypto-js/sha256';
import crypto from 'crypto';

import { createError } from 'api/utils';
import random from 'shared/uniqueID';
Expand All @@ -15,16 +14,15 @@ import mailer from '../utils/mailer';
import model from './usersModel';
import passwordRecoveriesModel from './passwordRecoveriesModel';
import settings from '../settings/settings';
import { generateUnlockCode } from './generateUnlockCode';

const MAX_FAILED_LOGIN_ATTEMPTS = 6;

const generateUnlockCode = () => crypto.randomBytes(32).toString('hex');

function conformRecoverText(options, _settings, domain, key, user) {
const response = {};
if (!options.newUser) {
response.subject = 'Password set';
response.text = `To set your password click on the following link:\n${domain}/setpassword/${key}`;
response.text = `To set your password click on the following link:\n${domain}/setpassword/${key}\nThis link will be valid for 24 hours.`;
}

if (options.newUser) {
Expand All @@ -34,7 +32,8 @@ function conformRecoverText(options, _settings, domain, key, user) {
`The administrators of ${siteName} have created an account for you under the user name:\n` +
`${user.username}\n\n` +
'To complete this process, please create a strong password by clicking on the following link:\n' +
`${domain}/setpassword/${key}?createAccount=true\n\n` +
`${domain}/setpassword/${key}?createAccount=true\n` +
'This link will be valid for 24 hours.\n\n' +
'For more information about the Uwazi platform, visit https://www.uwazi.io.\n\nThank you!\nUwazi team';

const htmlLink = `<a href="${domain}/setpassword/${key}?createAccount=true">${domain}/setpassword/${key}?createAccount=true</a>`;
Expand Down Expand Up @@ -288,7 +287,7 @@ export default {
},

recoverPassword(email, domain, options = {}) {
const key = SHA256(email + Date.now()).toString();
const key = generateUnlockCode();
return Promise.all([model.get({ email }), settings.get()]).then(([_user, _settings]) => {
const user = _user[0];
if (user) {
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "uwazi",
"version": "1.168.0-rc4",
"version": "1.168.0-rc5",
"description": "Uwazi is a free, open-source solution for organising, analysing and publishing your documents.",
"keywords": [
"react"
Expand Down

0 comments on commit 21c41c2

Please sign in to comment.