Skip to content
Snippets Groups Projects
Commit 26978f06 authored by Bastien DUMONT's avatar Bastien DUMONT :angel:
Browse files

Merge branch 'test/tempUsers' into 'dev'

test(temp-user): increase coverage

See merge request !394
parents d8674318 ccf7db3f
No related branches found
No related tags found
2 merge requests!404V3.1.0 (sans impression),!394test(temp-user): increase coverage
...@@ -149,6 +149,7 @@ export class StructuresController { ...@@ -149,6 +149,7 @@ export class StructuresController {
@Post(':id/claim') @Post(':id/claim')
public async claim(@Param('id') idStructure: string, @Body() user: User): Promise<pendingStructuresLink[]> { public async claim(@Param('id') idStructure: string, @Body() user: User): Promise<pendingStructuresLink[]> {
this.logger.debug(`claim structure ${idStructure} from ${user.email}`);
const structure = await this.structureService.findOne(idStructure); const structure = await this.structureService.findOne(idStructure);
return this.userService.updateStructureLinkedClaim(user.email, idStructure, structure); return this.userService.updateStructureLinkedClaim(user.email, idStructure, structure);
} }
...@@ -233,7 +234,7 @@ export class StructuresController { ...@@ -233,7 +234,7 @@ export class StructuresController {
@Param('id') id: string, @Param('id') id: string,
@Body() user: CreateTempUserDto @Body() user: CreateTempUserDto
): Promise<Types.ObjectId[] | TempUser> { ): Promise<Types.ObjectId[] | TempUser> {
// Get structure name this.logger.debug(`addOwner`);
const structure = await this.structureService.findOne(id); const structure = await this.structureService.findOne(id);
if (!structure) { if (!structure) {
throw new HttpException('Invalid Structure', HttpStatus.NOT_FOUND); throw new HttpException('Invalid Structure', HttpStatus.NOT_FOUND);
......
...@@ -27,26 +27,23 @@ describe('TempUserService', () => { ...@@ -27,26 +27,23 @@ describe('TempUserService', () => {
expect(tempUserController).toBeDefined(); expect(tempUserController).toBeDefined();
}); });
describe('findAll', () => {
it('should get all temporary users', async () => {
await tempUserController.findAll();
expect(mockTempUserService.findAll).toHaveBeenCalled();
});
});
describe('getTempUser', () => { describe('getTempUser', () => {
it('should get a temporary user by ID', async () => { it('should get a temporary user by ID', async () => {
const tmpUser = { email: 'test@test.com', structuresLink: [] }; mockTempUserService.findById.mockResolvedValueOnce('something');
mockTempUserService.findById.mockReturnValueOnce(tmpUser); await tempUserController.getTempUser('Azerty651');
expect(await tempUserController.getTempUser('addq651')).toEqual(tmpUser); expect(mockTempUserService.findById).toHaveBeenCalledWith('Azerty651');
});
it('should get all temporary users', async () => {
const tempUsers = [
{ email: 'test@test.com', structuresLink: [] },
{ email: 'test2@test.com', structuresLink: [] },
];
mockTempUserService.findAll.mockReturnValueOnce(tempUsers);
const findAllTempUsers = await tempUserController.findAll();
expect(findAllTempUsers.length).toBe(2);
}); });
it('should throw error in case of no users', async () => { it('should throw error in case of no users', async () => {
const tmpUser = null; mockTempUserService.findById.mockReturnValueOnce(null);
mockTempUserService.findById.mockReturnValueOnce(tmpUser);
try { try {
await tempUserController.getTempUser('addq651'); await tempUserController.getTempUser('Azerty651');
expect(true).toBe(false); expect(true).toBe(false);
} catch (e) { } catch (e) {
expect(e.message).toEqual('Temp user does not exist'); expect(e.message).toEqual('Temp user does not exist');
......
import { Controller, Get, HttpException, HttpStatus, Param } from '@nestjs/common'; import { Controller, Get, HttpException, HttpStatus, Param } from '@nestjs/common';
import { ApiParam } from '@nestjs/swagger'; import { ApiParam, ApiTags } from '@nestjs/swagger';
import { TempUser } from './temp-user.schema'; import { TempUser } from './temp-user.schema';
import { TempUserService } from './temp-user.service'; import { TempUserService } from './temp-user.service';
@ApiTags('temp-user')
@Controller('temp-user') @Controller('temp-user')
export class TempUserController { export class TempUserController {
constructor(private readonly tempUserService: TempUserService) {} constructor(private readonly tempUserService: TempUserService) {}
@Get() @Get()
public async findAll(): Promise<TempUser[]> { public async findAll(): Promise<TempUser[]> {
try { return await this.tempUserService.findAll();
const tempUsers: TempUser[] = await this.tempUserService.findAll();
return tempUsers;
} catch (error) {
throw new HttpException('An error occurred while fetching temp users.', HttpStatus.INTERNAL_SERVER_ERROR);
}
} }
@Get(':id') @Get(':id')
......
...@@ -7,6 +7,7 @@ import { MailerModule } from '../mailer/mailer.module'; ...@@ -7,6 +7,7 @@ import { MailerModule } from '../mailer/mailer.module';
import { MailerService } from '../mailer/mailer.service'; import { MailerService } from '../mailer/mailer.service';
import { TempUserService } from './temp-user.service'; import { TempUserService } from './temp-user.service';
import { Types } from 'mongoose'; import { Types } from 'mongoose';
import { CreateTempUserDto } from './dto/create-temp-user.dto';
const ejsSpy = jest.spyOn(ejs, 'renderFile'); const ejsSpy = jest.spyOn(ejs, 'renderFile');
...@@ -15,14 +16,15 @@ describe('TempUserService', () => { ...@@ -15,14 +16,15 @@ describe('TempUserService', () => {
const tempUserModelMock = { const tempUserModelMock = {
create: jest.fn(), create: jest.fn(),
findOne: jest.fn(), findOne: jest.fn().mockReturnThis(),
findById: jest.fn(), findById: jest.fn(),
findByIdAndUpdate: jest.fn(), findByIdAndUpdate: jest.fn(),
deleteOne: jest.fn(), deleteOne: jest.fn().mockReturnThis(),
find: jest.fn().mockReturnThis(), find: jest.fn().mockReturnThis(),
exec: jest.fn(), exec: jest.fn(),
updateOne: jest.fn(), updateOne: jest.fn(),
populate: jest.fn().mockReturnThis(), populate: jest.fn().mockReturnThis(),
select: jest.fn(),
}; };
const mockMailService = { const mockMailService = {
...@@ -43,7 +45,6 @@ describe('TempUserService', () => { ...@@ -43,7 +45,6 @@ describe('TempUserService', () => {
}, },
}; };
// Set up the test module before each test
beforeEach(async () => { beforeEach(async () => {
const module = await Test.createTestingModule({ const module = await Test.createTestingModule({
imports: [HttpModule, MailerModule], imports: [HttpModule, MailerModule],
...@@ -60,34 +61,38 @@ describe('TempUserService', () => { ...@@ -60,34 +61,38 @@ describe('TempUserService', () => {
tempUserService = module.get<TempUserService>(TempUserService); tempUserService = module.get<TempUserService>(TempUserService);
}); });
// Test whether the service is defined
it('should be defined', () => { it('should be defined', () => {
expect(tempUserService).toBeDefined(); expect(tempUserService).toBeDefined();
}); });
// Test related to the 'findAll' method
describe('findAll', () => { describe('findAll', () => {
it('should return a list of temp users', async () => { it('should return a list of temp users', async () => {
const tempUsers = [ const tempUsers = [
{ email: 'user1@test.com', structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')] }, { email: 'user1@test.com', structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')] },
{ email: 'user2@test.com', structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')] }, { email: 'user2@test.com', structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')] },
]; ];
tempUserModelMock.exec.mockResolvedValueOnce(tempUsers); tempUserModelMock.exec.mockResolvedValueOnce(tempUsers);
expect(await tempUserService.findAll()).toEqual(tempUsers); expect(await tempUserService.findAll()).toEqual(tempUsers);
}); });
it('should throw and error', async () => {
tempUserModelMock.exec.mockRejectedValueOnce(new Error('test'));
try {
await tempUserService.findAll();
expect(true).toBe(false);
} catch (e) {
expect(e.message).toEqual('An error occurred while fetching temp users.');
expect(e.status).toEqual(HttpStatus.INTERNAL_SERVER_ERROR);
}
});
}); });
// Tests related to the 'create' method
describe('create', () => { describe('create', () => {
const tmpUser = { const tmpUser: CreateTempUserDto = {
email: 'test@test.com', email: 'test@test.com',
structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')], structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')],
}; };
it('should not create temporary user if it already exists', async () => { it('should not create temporary user if it already exists', async () => {
tempUserModelMock.findOne.mockReturnThis();
tempUserModelMock.exec.mockResolvedValueOnce(tmpUser); tempUserModelMock.exec.mockResolvedValueOnce(tmpUser);
try { try {
await tempUserService.create(tmpUser, 'PIMMS Framboise'); await tempUserService.create(tmpUser, 'PIMMS Framboise');
...@@ -99,7 +104,6 @@ describe('TempUserService', () => { ...@@ -99,7 +104,6 @@ describe('TempUserService', () => {
}); });
it('should create a temporary user', async () => { it('should create a temporary user', async () => {
tempUserModelMock.findOne.mockReturnThis();
tempUserModelMock.exec.mockResolvedValueOnce(null); tempUserModelMock.exec.mockResolvedValueOnce(null);
tempUserModelMock.create.mockResolvedValueOnce(tmpUser); tempUserModelMock.create.mockResolvedValueOnce(tmpUser);
tempUserModelMock.exec.mockResolvedValueOnce(tmpUser); tempUserModelMock.exec.mockResolvedValueOnce(tmpUser);
...@@ -109,18 +113,15 @@ describe('TempUserService', () => { ...@@ -109,18 +113,15 @@ describe('TempUserService', () => {
}); });
}); });
// Test whether the 'findOne' method works
it('should find one', async () => { it('should find one', async () => {
const tmpUser = { const tmpUser = {
email: 'test2@test.com', email: 'test2@test.com',
structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')], structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')],
}; };
tempUserModelMock.findOne.mockReturnThis();
tempUserModelMock.exec.mockResolvedValueOnce(tmpUser); tempUserModelMock.exec.mockResolvedValueOnce(tmpUser);
expect(await tempUserService.findOne('test2@test.com')).toEqual(tmpUser); expect(await tempUserService.findOne('test2@test.com')).toEqual(tmpUser);
}); });
// Test whether the 'findById' method works
it('should find one by id', async () => { it('should find one by id', async () => {
const tmpUser = { const tmpUser = {
email: 'test2@test.com', email: 'test2@test.com',
...@@ -131,22 +132,18 @@ describe('TempUserService', () => { ...@@ -131,22 +132,18 @@ describe('TempUserService', () => {
expect(await tempUserService.findById('5fbb92e480a5c257dc0161f0')).toEqual(tmpUser); expect(await tempUserService.findById('5fbb92e480a5c257dc0161f0')).toEqual(tmpUser);
}); });
// Tests related to the 'delete' method
describe('delete', () => { describe('delete', () => {
it('should delete a temporary user', async () => { it('should delete a temporary user', async () => {
const tmpUser = { const tmpUser = {
email: 'test2@test.com', email: 'test2@test.com',
structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')], structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')],
}; };
tempUserModelMock.findOne.mockReturnThis();
tempUserModelMock.exec.mockResolvedValueOnce(tmpUser); tempUserModelMock.exec.mockResolvedValueOnce(tmpUser);
tempUserModelMock.deleteOne.mockReturnThis();
tempUserModelMock.exec.mockImplementationOnce(() => ({})); tempUserModelMock.exec.mockImplementationOnce(() => ({}));
expect(await tempUserService.delete('toto@test.com')).toEqual(tmpUser); expect(await tempUserService.delete('toto@test.com')).toEqual(tmpUser);
}); });
it('should return an error if the user does not exist', async () => { it('should return an error if the user does not exist', async () => {
tempUserModelMock.findOne.mockReturnThis();
tempUserModelMock.exec.mockResolvedValueOnce(null); tempUserModelMock.exec.mockResolvedValueOnce(null);
try { try {
await tempUserService.delete('toto@test.com'); await tempUserService.delete('toto@test.com');
...@@ -158,38 +155,53 @@ describe('TempUserService', () => { ...@@ -158,38 +155,53 @@ describe('TempUserService', () => {
}); });
}); });
// Tests related to the 'updateStructureLinked' method
describe('updateStructureLinked', () => { describe('updateStructureLinked', () => {
it('should update the linked structure', async () => { it('should not update the linked structure if the user is not found', async () => {
const tmpUser = { const createTempUserDto = {
email: 'test2@test.com', email: 'test2@test.com',
structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')], structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')],
}; };
tempUserModelMock.findOne.mockReturnThis(); tempUserModelMock.exec.mockResolvedValueOnce(null);
tempUserModelMock.exec.mockResolvedValueOnce(tmpUser);
const userInDb = { try {
structuresLink: [], await tempUserService.updateStructureLinked(createTempUserDto);
expect(true).toBe(false);
} catch (e) {
expect(e.message).toEqual('User not found');
expect(e.status).toEqual(HttpStatus.NOT_FOUND);
}
});
it('should not update the linked structure if the user is already linked', async () => {
const tmpUser = {
email: 'test2@test.com',
structuresLink: ['653015f722f6c63019ab038d'],
} as any;
const existingUserInDb = {
structuresLink: ['653015f722f6c63019ab038d', '653015f722f6c63019ab038e'],
save: jest.fn().mockResolvedValueOnce(tmpUser), save: jest.fn().mockResolvedValueOnce(tmpUser),
}; };
jest.spyOn(tempUserModelMock, 'findOne').mockReturnValue({ tempUserModelMock.exec.mockResolvedValueOnce(existingUserInDb);
exec: jest.fn().mockResolvedValueOnce(userInDb),
});
tempUserModelMock.findByIdAndUpdate.mockReturnThis(); try {
expect(await tempUserService.updateStructureLinked(tmpUser)).toEqual(tmpUser); await tempUserService.updateStructureLinked(tmpUser);
expect(true).toBe(false);
} catch (e) {
expect(e.message).toEqual('User already linked');
expect(e.status).toEqual(HttpStatus.UNPROCESSABLE_ENTITY);
}
}); });
it('should update the linked structure', async () => {
it('should not update the linked structure if the user is already linked', async () => {
const tmpUser = { const tmpUser = {
email: 'test2@test.com', email: 'test2@test.com',
structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')], structuresLink: [new Types.ObjectId('653015f722f6c63019ab038d')],
}; };
tempUserModelMock.findOne.mockReturnThis(); tempUserModelMock.findOne.mockReturnThis();
tempUserModelMock.exec.mockResolvedValueOnce(tmpUser);
const userInDb = { const userInDb = {
structuresLink: [new Types.ObjectId()], structuresLink: [],
save: jest.fn().mockResolvedValueOnce(tmpUser), save: jest.fn().mockResolvedValueOnce(tmpUser),
}; };
...@@ -198,12 +210,41 @@ describe('TempUserService', () => { ...@@ -198,12 +210,41 @@ describe('TempUserService', () => {
}); });
tempUserModelMock.findByIdAndUpdate.mockReturnThis(); tempUserModelMock.findByIdAndUpdate.mockReturnThis();
expect(await tempUserService.updateStructureLinked(tmpUser)).toEqual(tmpUser);
});
});
describe('getStructureTempUsers', () => {
it('should return a list of temp users', async () => {
tempUserModelMock.select.mockReturnThis();
await tempUserService.getStructureTempUsers('653015f722f6c63019ab038d');
expect(tempUserModelMock.find).toHaveBeenCalled();
});
});
describe('removeFromStructureLinked', () => {
it('should not find the user', async () => {
tempUserModelMock.findOne.mockReturnThis();
tempUserModelMock.exec.mockResolvedValueOnce(null);
try { try {
await tempUserService.updateStructureLinked(tmpUser); await tempUserService.removeFromStructureLinked('jp@test.com', '653015f722f6c63019ab038d');
} catch (e) { } catch (e) {
expect(e.message).toEqual('User is already linked'); expect(e.status).toEqual(HttpStatus.NOT_FOUND);
expect(e.status).toEqual(HttpStatus.UNPROCESSABLE_ENTITY); expect(e.message).toEqual('Invalid temp user');
}
});
it('should fail because temp user is not linked to the structure', async () => {
const tmpUser = {
email: 'test2@test.com',
structuresLink: [new Types.ObjectId('aaa015f722f6c63019ab038d')],
};
tempUserModelMock.findOne.mockReturnThis();
tempUserModelMock.exec.mockResolvedValueOnce(tmpUser);
try {
await tempUserService.removeFromStructureLinked('jp@test.com', '653015f722f6c63019ab038d');
} catch (e) {
expect(e.status).toEqual(HttpStatus.NOT_FOUND);
expect(e.message).toEqual("Temp user doesn't belong to this structure");
} }
}); });
}); });
......
...@@ -23,13 +23,13 @@ export class TempUserService { ...@@ -23,13 +23,13 @@ export class TempUserService {
} }
const createUser = await this.tempUserModel.create(createTempUser); const createUser = await this.tempUserModel.create(createTempUser);
this.logger.debug(`TempUsersService | tempUser created`); this.logger.debug('tempUser created');
// Send email
this.sendUserMail(createUser, structureName); this.sendUserMail(createUser, structureName);
return this.findOne(createTempUser.email); return this.findOne(createTempUser.email);
} }
public async findAll(): Promise<ITempUser[]> { public async findAll(): Promise<ITempUser[]> {
this.logger.debug('findAll');
try { try {
const tempUsers = await this.tempUserModel const tempUsers = await this.tempUserModel
.find() .find()
...@@ -41,19 +41,22 @@ export class TempUserService { ...@@ -41,19 +41,22 @@ export class TempUserService {
return tempUsers; return tempUsers;
} catch (error) { } catch (error) {
throw new Error('An error occurred while fetching temp users: ' + error.message); throw new HttpException('An error occurred while fetching temp users.', HttpStatus.INTERNAL_SERVER_ERROR);
} }
} }
public async findOne(mail: string): Promise<ITempUser> { public async findOne(mail: string): Promise<ITempUser> {
this.logger.debug('findOne');
return this.tempUserModel.findOne({ email: mail }).exec(); return this.tempUserModel.findOne({ email: mail }).exec();
} }
public async findById(id: string): Promise<TempUser> { public async findById(id: string): Promise<TempUser> {
this.logger.debug('findById');
return this.tempUserModel.findById(new Types.ObjectId(id)).exec(); return this.tempUserModel.findById(new Types.ObjectId(id)).exec();
} }
public async delete(mail: string): Promise<TempUser> { public async delete(mail: string): Promise<TempUser> {
this.logger.debug('delete');
const userInDb = await this.findOne(mail); const userInDb = await this.findOne(mail);
if (!userInDb) { if (!userInDb) {
throw new HttpException('User does not exist', HttpStatus.BAD_REQUEST); throw new HttpException('User does not exist', HttpStatus.BAD_REQUEST);
...@@ -63,24 +66,19 @@ export class TempUserService { ...@@ -63,24 +66,19 @@ export class TempUserService {
} }
public async updateStructureLinked(createTempUser: CreateTempUserDto): Promise<TempUser> { public async updateStructureLinked(createTempUser: CreateTempUserDto): Promise<TempUser> {
// Find the user by email this.logger.debug('updateStructureLinked');
const userInDb = await this.tempUserModel.findOne({ email: createTempUser.email }).exec(); const user = await this.tempUserModel.findOne({ email: createTempUser.email }).exec();
if (!user) {
if (!userInDb) {
throw new HttpException('User not found', HttpStatus.NOT_FOUND); throw new HttpException('User not found', HttpStatus.NOT_FOUND);
} }
// Check if the structuresLink already contains the provided structure // Check if the structuresLink already contains the provided structure
if (userInDb.structuresLink.includes(createTempUser.structuresLink[0])) { if (user.structuresLink.includes(createTempUser.structuresLink[0])) {
throw new HttpException('User already linked', HttpStatus.UNPROCESSABLE_ENTITY); throw new HttpException('User already linked', HttpStatus.UNPROCESSABLE_ENTITY);
} }
// Add the new structure to structuresLink // Add the new structure to structuresLink
userInDb.structuresLink.push(createTempUser.structuresLink[0]); user.structuresLink.push(createTempUser.structuresLink[0]);
const updatedUser = await user.save();
// Save the updated user
const updatedUser = await userInDb.save();
return updatedUser; return updatedUser;
} }
...@@ -134,7 +132,6 @@ export class TempUserService { ...@@ -134,7 +132,6 @@ export class TempUserService {
// Save the updated user if there are remaining structures // Save the updated user if there are remaining structures
await user.save(); await user.save();
} }
return user.structuresLink; return user.structuresLink;
} }
} }
...@@ -504,10 +504,10 @@ export class UsersService { ...@@ -504,10 +504,10 @@ export class UsersService {
idStructure: string, idStructure: string,
structure: StructureDocument structure: StructureDocument
): Promise<pendingStructuresLink[]> { ): Promise<pendingStructuresLink[]> {
const stucturesLinked = this.updatePendingStructureLinked(userEmail, idStructure, structure.structureName); this.logger.debug(`updateStructureLinkedClaim for structure ${idStructure} from ${userEmail}`);
const structuresLinked = this.updatePendingStructureLinked(userEmail, idStructure, structure.structureName);
this.sendAdminStructureValidationMail(userEmail, structure); this.sendAdminStructureValidationMail(userEmail, structure);
return structuresLinked;
return stucturesLinked;
} }
/** /**
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment