feat: implement MailModule with email services and tests
Added MailModule with services for email validation and password reset functionalities. Includes configuration via `@nestjs-modules/mailer` and comprehensive unit tests.
This commit is contained in:
29
backend/src/mail/mail.module.ts
Normal file
29
backend/src/mail/mail.module.ts
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import { Module } from "@nestjs/common";
|
||||||
|
import { ConfigService } from "@nestjs/config";
|
||||||
|
import { MailerModule } from "@nestjs-modules/mailer";
|
||||||
|
import { MailService } from "./mail.service";
|
||||||
|
|
||||||
|
@Module({
|
||||||
|
imports: [
|
||||||
|
MailerModule.forRootAsync({
|
||||||
|
useFactory: async (config: ConfigService) => ({
|
||||||
|
transport: {
|
||||||
|
host: config.get("MAIL_HOST"),
|
||||||
|
port: Number(config.get("MAIL_PORT")),
|
||||||
|
secure: config.get("MAIL_SECURE") === "true",
|
||||||
|
auth: {
|
||||||
|
user: config.get("MAIL_USER"),
|
||||||
|
pass: config.get("MAIL_PASS"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaults: {
|
||||||
|
from: `"No Reply" <${config.get("MAIL_FROM")}>`,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
inject: [ConfigService],
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
providers: [MailService],
|
||||||
|
exports: [MailService],
|
||||||
|
})
|
||||||
|
export class MailModule {}
|
||||||
87
backend/src/mail/mail.service.spec.ts
Normal file
87
backend/src/mail/mail.service.spec.ts
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
import { Logger } from "@nestjs/common";
|
||||||
|
import { ConfigService } from "@nestjs/config";
|
||||||
|
import { Test, TestingModule } from "@nestjs/testing";
|
||||||
|
import { MailerService } from "@nestjs-modules/mailer";
|
||||||
|
import { MailService } from "./mail.service";
|
||||||
|
|
||||||
|
describe("MailService", () => {
|
||||||
|
let service: MailService;
|
||||||
|
let mailerService: MailerService;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const module: TestingModule = await Test.createTestingModule({
|
||||||
|
providers: [
|
||||||
|
MailService,
|
||||||
|
{
|
||||||
|
provide: MailerService,
|
||||||
|
useValue: {
|
||||||
|
sendMail: jest.fn().mockResolvedValue({}),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
provide: ConfigService,
|
||||||
|
useValue: {
|
||||||
|
get: jest.fn().mockReturnValue("test.io"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}).compile();
|
||||||
|
|
||||||
|
service = module.get<MailService>(MailService);
|
||||||
|
mailerService = module.get<MailerService>(MailerService);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be defined", () => {
|
||||||
|
expect(service).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should send validation email with correct URL", async () => {
|
||||||
|
const email = "test@example.com";
|
||||||
|
const token = "token123";
|
||||||
|
await service.sendEmailValidation(email, token);
|
||||||
|
|
||||||
|
expect(mailerService.sendMail).toHaveBeenCalledWith(
|
||||||
|
expect.objectContaining({
|
||||||
|
to: email,
|
||||||
|
subject: "Validation de votre adresse email",
|
||||||
|
html: expect.stringContaining(
|
||||||
|
"https://test.io/api/auth/verify-email?token=token123",
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should send password reset email with correct URL", async () => {
|
||||||
|
const email = "test@example.com";
|
||||||
|
const token = "token123";
|
||||||
|
await service.sendPasswordReset(email, token);
|
||||||
|
|
||||||
|
expect(mailerService.sendMail).toHaveBeenCalledWith(
|
||||||
|
expect.objectContaining({
|
||||||
|
to: email,
|
||||||
|
subject: "Réinitialisation de votre mot de passe",
|
||||||
|
html: expect.stringContaining(
|
||||||
|
"https://test.io/api/auth/reset-password?token=token123",
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should throw and log error when sending fails", async () => {
|
||||||
|
const email = "test@example.com";
|
||||||
|
const token = "token123";
|
||||||
|
const error = new Error("SMTP Error");
|
||||||
|
(mailerService.sendMail as jest.Mock).mockRejectedValueOnce(error);
|
||||||
|
|
||||||
|
const loggerSpy = jest
|
||||||
|
.spyOn(Logger.prototype, "error")
|
||||||
|
.mockImplementation(() => {});
|
||||||
|
|
||||||
|
await expect(service.sendEmailValidation(email, token)).rejects.toThrow(
|
||||||
|
"SMTP Error",
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(loggerSpy).toHaveBeenCalled();
|
||||||
|
loggerSpy.mockRestore();
|
||||||
|
});
|
||||||
|
});
|
||||||
62
backend/src/mail/mail.service.ts
Normal file
62
backend/src/mail/mail.service.ts
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
import { Injectable, Logger } from "@nestjs/common";
|
||||||
|
import { ConfigService } from "@nestjs/config";
|
||||||
|
import { MailerService } from "@nestjs-modules/mailer";
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class MailService {
|
||||||
|
private readonly logger = new Logger(MailService.name);
|
||||||
|
private readonly domain: string;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
private mailerService: MailerService,
|
||||||
|
private configService: ConfigService,
|
||||||
|
) {
|
||||||
|
this.domain = this.configService.get<string>("DOMAIN_NAME", "memegoat.io");
|
||||||
|
}
|
||||||
|
|
||||||
|
async sendEmailValidation(email: string, token: string) {
|
||||||
|
const url = `https://${this.domain}/api/auth/verify-email?token=${token}`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.mailerService.sendMail({
|
||||||
|
to: email,
|
||||||
|
subject: "Validation de votre adresse email",
|
||||||
|
html: `
|
||||||
|
<p>Bonjour,</p>
|
||||||
|
<p>Veuillez cliquer sur le lien ci-dessous pour valider votre adresse email :</p>
|
||||||
|
<p><a href="${url}">${url}</a></p>
|
||||||
|
<p>Si vous n'avez pas demandé cette validation, vous pouvez ignorer cet email.</p>
|
||||||
|
`,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
this.logger.error(
|
||||||
|
`Failed to send validation email to ${email}: ${error.message}`,
|
||||||
|
error.stack,
|
||||||
|
);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async sendPasswordReset(email: string, token: string) {
|
||||||
|
const url = `https://${this.domain}/api/auth/reset-password?token=${token}`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.mailerService.sendMail({
|
||||||
|
to: email,
|
||||||
|
subject: "Réinitialisation de votre mot de passe",
|
||||||
|
html: `
|
||||||
|
<p>Bonjour,</p>
|
||||||
|
<p>Veuillez cliquer sur le lien ci-dessous pour réinitialiser votre mot de passe :</p>
|
||||||
|
<p><a href="${url}">${url}</a></p>
|
||||||
|
<p>Si vous n'avez pas demandé de réinitialisation, vous pouvez ignorer cet email.</p>
|
||||||
|
`,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
this.logger.error(
|
||||||
|
`Failed to send password reset email to ${email}: ${error.message}`,
|
||||||
|
error.stack,
|
||||||
|
);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user