mirror of
https://github.com/hexastack/hexabot
synced 2025-05-03 20:41:15 +00:00
156 lines
5.8 KiB
TypeScript
156 lines
5.8 KiB
TypeScript
/*
|
|
* Copyright © 2024 Hexastack. All rights reserved.
|
|
*
|
|
* Licensed under the GNU Affero General Public License v3.0 (AGPLv3) with the following additional terms:
|
|
* 1. The name "Hexabot" is a trademark of Hexastack. You may not use this name in derivative works without express written permission.
|
|
* 2. All derivative works must include clear attribution to the original creator and software, Hexastack and Hexabot, in a prominent location (e.g., in the software's "About" section, documentation, and README file).
|
|
* 3. SaaS Restriction: This software, or any derivative of it, may not be used to offer a competing product or service (SaaS) without prior written consent from Hexastack. Offering the software as a service or using it in a commercial cloud environment without express permission is strictly prohibited.
|
|
*/
|
|
|
|
import { CACHE_MANAGER } from '@nestjs/cache-manager';
|
|
import { NotFoundException } from '@nestjs/common';
|
|
import { EventEmitter2 } from '@nestjs/event-emitter';
|
|
import { JwtModule, JwtService } from '@nestjs/jwt';
|
|
import { MongooseModule, getModelToken } from '@nestjs/mongoose';
|
|
import { Test, TestingModule } from '@nestjs/testing';
|
|
import { ISendMailOptions, MailerService } from '@nestjs-modules/mailer';
|
|
import { compareSync } from 'bcryptjs';
|
|
import { Model } from 'mongoose';
|
|
import { SentMessageInfo } from 'nodemailer';
|
|
|
|
import { AttachmentRepository } from '@/attachment/repositories/attachment.repository';
|
|
import { AttachmentModel } from '@/attachment/schemas/attachment.schema';
|
|
import { AttachmentService } from '@/attachment/services/attachment.service';
|
|
import { LanguageRepository } from '@/i18n/repositories/language.repository';
|
|
import { LanguageModel } from '@/i18n/schemas/language.schema';
|
|
import { I18nService } from '@/i18n/services/i18n.service';
|
|
import { LanguageService } from '@/i18n/services/language.service';
|
|
import { LoggerService } from '@/logger/logger.service';
|
|
import { installUserFixtures, users } from '@/utils/test/fixtures/user';
|
|
import {
|
|
closeInMongodConnection,
|
|
rootMongooseTestModule,
|
|
} from '@/utils/test/test';
|
|
|
|
import { PasswordResetService } from './passwordReset.service';
|
|
import { RoleService } from './role.service';
|
|
import { UserService } from './user.service';
|
|
import { RoleRepository } from '../repositories/role.repository';
|
|
import { UserRepository } from '../repositories/user.repository';
|
|
import { PermissionModel } from '../schemas/permission.schema';
|
|
import { RoleModel } from '../schemas/role.schema';
|
|
import { User, UserModel } from '../schemas/user.schema';
|
|
|
|
describe('PasswordResetService', () => {
|
|
let passwordResetService: PasswordResetService;
|
|
let mailerService: MailerService;
|
|
let jwtService: JwtService;
|
|
let userModel: Model<User>;
|
|
beforeAll(async () => {
|
|
const module: TestingModule = await Test.createTestingModule({
|
|
imports: [
|
|
rootMongooseTestModule(installUserFixtures),
|
|
MongooseModule.forFeature([
|
|
UserModel,
|
|
RoleModel,
|
|
PermissionModel,
|
|
AttachmentModel,
|
|
LanguageModel,
|
|
]),
|
|
JwtModule,
|
|
],
|
|
providers: [
|
|
UserService,
|
|
UserRepository,
|
|
RoleService,
|
|
AttachmentService,
|
|
AttachmentRepository,
|
|
RoleRepository,
|
|
LanguageService,
|
|
LanguageRepository,
|
|
LoggerService,
|
|
PasswordResetService,
|
|
JwtService,
|
|
{
|
|
provide: MailerService,
|
|
useValue: {
|
|
sendMail: jest.fn(
|
|
(_options: ISendMailOptions): Promise<SentMessageInfo> =>
|
|
Promise.resolve('Mail sent successfully'),
|
|
),
|
|
},
|
|
},
|
|
{
|
|
provide: I18nService,
|
|
useValue: {
|
|
t: jest.fn().mockImplementation((t) => t),
|
|
},
|
|
},
|
|
EventEmitter2,
|
|
{
|
|
provide: CACHE_MANAGER,
|
|
useValue: {
|
|
del: jest.fn(),
|
|
get: jest.fn(),
|
|
set: jest.fn(),
|
|
},
|
|
},
|
|
],
|
|
}).compile();
|
|
passwordResetService =
|
|
module.get<PasswordResetService>(PasswordResetService);
|
|
|
|
mailerService = module.get<MailerService>(MailerService);
|
|
jwtService = module.get<JwtService>(JwtService);
|
|
userModel = module.get<Model<User>>(getModelToken('User'));
|
|
});
|
|
afterAll(async () => {
|
|
await closeInMongodConnection();
|
|
});
|
|
|
|
afterEach(jest.clearAllMocks);
|
|
describe('requestReset', () => {
|
|
it('should send an email with a token', async () => {
|
|
const sendMailSpy = jest.spyOn(mailerService, 'sendMail');
|
|
const signSpy = jest.spyOn(passwordResetService, 'sign');
|
|
const promise = passwordResetService.requestReset({
|
|
email: users[0].email,
|
|
});
|
|
await expect(promise).resolves.toBeUndefined();
|
|
|
|
expect(sendMailSpy).toHaveBeenCalled();
|
|
expect(signSpy).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should throw a 404 error', async () => {
|
|
const promise = passwordResetService.requestReset({
|
|
email: 'a@b.ca',
|
|
});
|
|
await expect(promise).rejects.toThrow(NotFoundException);
|
|
});
|
|
|
|
it('should return change the password', async () => {
|
|
const spy = jest.spyOn(passwordResetService, 'sign');
|
|
const verifySpy = jest.spyOn(passwordResetService, 'verify');
|
|
const token = jwtService.sign(
|
|
{ email: users[0].email },
|
|
passwordResetService.jwtSignOptions,
|
|
);
|
|
spy.mockResolvedValue(token);
|
|
await passwordResetService.requestReset({ email: users[0].email });
|
|
|
|
const promise = passwordResetService.reset(
|
|
{ password: 'newPassword' },
|
|
token,
|
|
);
|
|
|
|
await expect(promise).resolves.toBeUndefined();
|
|
expect(verifySpy).toHaveBeenCalled();
|
|
|
|
const user = await userModel.findOne({ email: users[0].email });
|
|
expect(user.resetToken).toBeNull();
|
|
expect(compareSync('newPassword', user.password)).toBeTruthy();
|
|
});
|
|
});
|
|
});
|