Compare commits

..

5 Commits

Author SHA1 Message Date
Omar Oughriss
5032fa3134 Move Mail feature to the main file 2025-07-18 16:25:35 +02:00
Omar Oughriss
8b6a25e174 Add Mailchimp feature 2025-07-18 14:41:46 +02:00
Omar Oughriss
d2b608ef94 Rename the main server just for now 2025-07-18 14:41:29 +02:00
Omar Oughriss
ef8c260b71 Add Mailchimp and dotenv deps 2025-07-18 14:41:07 +02:00
Omar Oughriss
dd4d6a5d94 Add .env exemple 2025-07-18 14:40:20 +02:00
3 changed files with 197 additions and 757 deletions

View File

@ -1,30 +1,8 @@
# Configuration OVH
OVH_APP_KEY=
OVH_APP_SECRET=
OVH_CONSUMER_KEY=
OVH_SMS_SERVICE_NAME=
# Configuration SMS Factor
SMS_FACTOR_TOKEN=
#Configuration Mailchimp #Configuration Mailchimp
MAILCHIMP_API_KEY= MAILCHIMP_API_KEY=
MAILCHIMP_KEY= MAILCHIMP_KEY=
MAILCHIMP_LIST_ID= MAILCHIMP_LIST_ID=
APP_HOST=
#Configuration Stripe
STRIPE_SECRET_KEY=
STRIPE_WEBHOOK_SECRET=
STRIPE_STANDARD_SUBSCRIPTION_PRICE_ID=
STRIPE_STANDARD_ANNUAL_SUBSCRIPTION_PRICE_ID=
STRIPE_UNLIMITED_SUBSCRIPTION_PRICE_ID=
STRIPE_UNLIMITED_ANNUAL_SUBSCRIPTION_PRICE_ID=
#Cartes de test Stripe
SUCCES= 4242 4242 4242 4242 #Paiement réussi
DECLINED= 4000 0025 0000 3155 #Paiement refusé
#Configuration serveur #Configuration serveur
APP_HOST=
PORT= PORT=

View File

@ -12,10 +12,7 @@
"cors": "^2.8.5", "cors": "^2.8.5",
"dotenv": "^17.2.0", "dotenv": "^17.2.0",
"express": "^4.18.2", "express": "^4.18.2",
"node-fetch": "^2.6.7", "node-fetch": "^2.6.7"
"ovh": "^2.0.3",
"stripe": "^18.3.0",
"uuid": "^11.1.0"
}, },
"devDependencies": { "devDependencies": {
"nodemon": "^3.0.1" "nodemon": "^3.0.1"

View File

@ -1,10 +1,7 @@
const express = require('express'); const express = require('express');
const cors = require('cors'); const cors = require('cors');
const fetch = require('node-fetch'); const fetch = require('node-fetch');
const { v4: uuidv4 } = require('uuid');
const ovh = require('ovh');
const mailchimp = require('@mailchimp/mailchimp_transactional'); const mailchimp = require('@mailchimp/mailchimp_transactional');
const Stripe = require('stripe');
require('dotenv').config(); require('dotenv').config();
// Initialisation de l'application Express // Initialisation de l'application Express
@ -13,7 +10,7 @@ const PORT = process.env.PORT || 8080;
// Configuration CORS // Configuration CORS
const corsOptions = { const corsOptions = {
origin: ['http://local.lecoffreio.4nkweb:3000', 'http://localhost:3000', 'https://lecoffreio.4nkweb.com'], origin: ['http://local.lecoffreio.4nkweb:3000', 'http://localhost:3000'],
methods: ['GET', 'POST', 'PUT', 'DELETE', 'OPTIONS'], methods: ['GET', 'POST', 'PUT', 'DELETE', 'OPTIONS'],
allowedHeaders: ['Content-Type', 'Authorization'] allowedHeaders: ['Content-Type', 'Authorization']
}; };
@ -21,99 +18,52 @@ const corsOptions = {
app.use(cors(corsOptions)); app.use(cors(corsOptions));
app.use(express.json()); app.use(express.json());
const authTokens = [];
const ECivility = {
MALE: 'MALE',
FEMALE: 'FEMALE',
OTHERS: 'OTHERS'
};
const EOfficeStatus = {
ACTIVATED: 'ACTIVATED',
DESACTIVATED: 'DESACTIVATED'
};
const EIdnotRole = {
DIRECTEUR: "Directeur général du CSN",
NOTAIRE_TITULAIRE: "Notaire titulaire",
NOTAIRE_ASSOCIE: "Notaire associé",
NOTAIRE_SALARIE: "Notaire salarié",
COLLABORATEUR: "Collaborateur",
SECRETAIRE_GENERAL: "Secrétaire général",
SUPPLEANT: "Suppléant",
ADMINISTRATEUR: "Administrateur",
RESPONSABLE: "Responsable",
CURATEUR: "Curateur",
}
function getOfficeStatus(statusName) { function getOfficeStatus(statusName) {
switch (statusName) { switch (statusName) {
case "Pourvu": case 'Pourvu':
return EOfficeStatus.ACTIVATED; return 'ACTIVATED';
case "Pourvu mais décédé": case 'Pourvu mais décédé':
return EOfficeStatus.ACTIVATED; return 'ACTIVATED';
case "Sans titulaire": case 'Sans titulaire':
return EOfficeStatus.ACTIVATED; return 'ACTIVATED';
case "Vacance": case 'Vacance':
return EOfficeStatus.ACTIVATED; return 'ACTIVATED';
case "En activité": case 'En activité':
return EOfficeStatus.ACTIVATED; return 'ACTIVATED';
default: default:
return EOfficeStatus.DESACTIVATED; return 'DESACTIVATED';
}
}
function getOfficeRole(roleName) {
switch (roleName) {
case EIdnotRole.NOTAIRE_TITULAIRE:
return { name: 'Notaire' };
case EIdnotRole.NOTAIRE_ASSOCIE:
return { name: 'Notaire' };
case EIdnotRole.NOTAIRE_SALARIE:
return { name: 'Notaire' };
case EIdnotRole.COLLABORATEUR:
return { name: 'Collaborateur' };
case EIdnotRole.SUPPLEANT:
return { name: 'Collaborateur' };
case EIdnotRole.ADMINISTRATEUR:
return { name: 'Collaborateur' };
case EIdnotRole.CURATEUR:
return { name: 'Collaborateur' };
default:
return null;
} }
} }
function getRole(roleName) { function getRole(roleName) {
switch (roleName) { switch (roleName) {
case EIdnotRole.NOTAIRE_TITULAIRE: case 'Notaire titulaire':
return { name: 'admin' }; return { name: 'admin', label: 'Administrateur' };
case EIdnotRole.NOTAIRE_ASSOCIE: case 'Notaire associé':
return { name: 'admin' }; return { name: 'admin', label: 'Administrateur' };
case EIdnotRole.NOTAIRE_SALARIE: case 'Notaire salarié':
return { name: 'notary' }; return { name: 'notary', label: 'Notaire' };
case EIdnotRole.COLLABORATEUR: case 'Collaborateur':
return { name: 'notary' }; return { name: 'notary', label: 'Notaire' };
case EIdnotRole.SUPPLEANT: case 'Suppléant':
return { name: 'notary' }; return { name: 'notary', label: 'Notaire' };
case EIdnotRole.ADMINISTRATEUR: case 'Administrateur':
return { name: 'admin' }; return { name: 'admin', label: 'Administrateur' };
case EIdnotRole.CURATEUR: case 'Curateur':
return { name: 'notary' }; return { name: 'notary', label: 'Notaire' };
default: default:
return { name: 'default' }; return { name: 'default', label: 'Défaut' };
} }
} }
function getCivility(civility) { function getCivility(civility) {
switch (civility) { switch (civility) {
case 'Monsieur': case 'Monsieur':
return ECivility.MALE; return 'MALE';
case 'Madame': case 'Madame':
return ECivility.FEMALE; return 'FEMALE';
default: default:
return ECivility.OTHERS; return 'OTHERS';
} }
} }
@ -187,7 +137,7 @@ app.post('/api/v1/idnot/user/:code', async (req, res) => {
return null; return null;
} }
const idNotUser = { const idnotUser = {
idNot: payload.sub, idNot: payload.sub,
office: { office: {
idNot: payload.entity_idn, idNot: payload.entity_idn,
@ -209,19 +159,15 @@ app.post('/api/v1/idnot/user/:code', async (req, res) => {
phone_number: userData.numeroTelephone, phone_number: userData.numeroTelephone,
cell_phone_number: userData.numeroMobile ?? userData.numeroTelephone, cell_phone_number: userData.numeroMobile ?? userData.numeroTelephone,
civility: getCivility(userData.personne.civilite) civility: getCivility(userData.personne.civilite)
}, }
office_role: getOfficeRole(userData.typeLien.name)
}; };
if (!idNotUser.contact.email) { if (!idnotUser.contact.email) {
console.error('User pro email empty'); console.error("User pro email empty");
return null; return null;
} }
const authToken = uuidv4(); res.json(idnotUser);
authTokens.push({ idNot: idNotUser.idNot, authToken });
res.json({ idNotUser, authToken });
} catch (error) { } catch (error) {
res.status(500).json({ res.status(500).json({
error: 'Internal Server Error', error: 'Internal Server Error',
@ -230,226 +176,10 @@ app.post('/api/v1/idnot/user/:code', async (req, res) => {
} }
}); });
//------------------------------------ SMS Section ----------------------------------------- //------------------------------------ Email Section ------------------------------------
const configSms = {
// OVH config
OVH_APP_KEY: process.env.OVH_APP_KEY,
OVH_APP_SECRET: process.env.OVH_APP_SECRET,
OVH_CONSUMER_KEY: process.env.OVH_CONSUMER_KEY,
OVH_SMS_SERVICE_NAME: process.env.OVH_SMS_SERVICE_NAME,
// SMS Factor config
SMS_FACTOR_TOKEN: process.env.SMS_FACTOR_TOKEN,
PORT: process.env.PORT || 8080
};
// Codes storage
const verificationCodes = new Map();
// Service SMS
class SmsService {
static generateCode() {
return Math.floor(100000 + Math.random() * 900000);
}
// OVH Service
static sendSmsWithOvh(phoneNumber, message) {
return new Promise((resolve, reject) => {
const ovhClient = ovh({
appKey: configSms.OVH_APP_KEY,
appSecret: configSms.OVH_APP_SECRET,
consumerKey: configSms.OVH_CONSUMER_KEY
});
ovhClient.request('POST', `/sms/${configSms.OVH_SMS_SERVICE_NAME}/jobs`, {
message: message,
receivers: [phoneNumber],
senderForResponse: false,
sender: 'not.IT Fact',
noStopClause: true
}, (error, result) => {
if (error) {
console.error('Erreur OVH SMS:', error);
resolve({ success: false, error: 'Échec de l\'envoi du SMS via OVH' });
} else {
resolve({ success: true });
}
});
});
}
// SMS Factor Service
static async sendSmsWithSmsFactor(phoneNumber, message) {
try {
const url = new URL('https://api.smsfactor.com/send/simulate');
url.searchParams.append('to', phoneNumber);
url.searchParams.append('text', message);
url.searchParams.append('sender', 'LeCoffre');
url.searchParams.append('token', configSms.SMS_FACTOR_TOKEN);
const response = await fetch(url.toString());
if (!response.ok) {
throw new Error(`HTTP error! status: ${response.status}`);
}
return { success: true };
} catch (error) {
console.error('Erreur SMS Factor:', error);
return { success: false, error: 'Échec de l\'envoi du SMS via SMS Factor' };
}
}
// Main method
static async sendSms(phoneNumber, message) {
// Try first with OVH
const ovhResult = await this.sendSmsWithOvh(phoneNumber, message);
if (ovhResult.success) {
return ovhResult;
}
// If OVH fails, try with SMS Factor
console.log('OVH SMS failed, trying SMS Factor...');
return await this.sendSmsWithSmsFactor(phoneNumber, message);
}
}
// Phone number validation middleware
const validatePhoneNumber = (req, res, next) => {
const { phoneNumber } = req.body;
if (!phoneNumber) {
return res.status(400).json({
success: false,
message: 'Le numéro de téléphone est requis'
});
}
// Validation basique du format
const phoneRegex = /^\+?[1-9]\d{1,14}$/;
if (!phoneRegex.test(phoneNumber)) {
return res.status(400).json({
success: false,
message: 'Format de numéro de téléphone invalide'
});
}
next();
};
// Routes
app.post('/api/send-code', validatePhoneNumber, async (req, res) => {
const { phoneNumber } = req.body;
try {
// Check if a code already exists and is not expired
const existingVerification = verificationCodes.get(phoneNumber);
if (existingVerification) {
const timeSinceLastSend = Date.now() - existingVerification.timestamp;
if (timeSinceLastSend < 30000) { // 30 secondes
return res.status(429).json({
success: false,
message: 'Veuillez attendre 30 secondes avant de demander un nouveau code'
});
}
}
// Generate a new code
const code = SmsService.generateCode();
// Store the code
verificationCodes.set(phoneNumber, {
code,
timestamp: Date.now(),
attempts: 0
});
// Send the SMS
const message = `Votre code de vérification LeCoffre est : ${code}`;
const result = await SmsService.sendSms(phoneNumber, message);
if (result.success) {
res.json({
success: true,
message: 'Code envoyé avec succès',
});
} else {
res.status(500).json({
success: false,
message: 'Échec de l\'envoi du SMS via les deux fournisseurs'
});
}
} catch (error) {
console.error('Erreur:', error);
res.status(500).json({
success: false,
message: 'Erreur serveur lors de l\'envoi du code'
});
}
});
app.post('/api/verify-code', validatePhoneNumber, (req, res) => {
const { phoneNumber, code } = req.body;
if (!code) {
return res.status(400).json({
success: false,
message: 'Le code est requis'
});
}
const verification = verificationCodes.get(phoneNumber);
if (!verification) {
return res.status(400).json({
success: false,
message: 'Aucun code n\'a été envoyé à ce numéro'
});
}
// Check if the code has not expired (5 minutes)
if (Date.now() - verification.timestamp > 5 * 60 * 1000) {
verificationCodes.delete(phoneNumber);
return res.status(400).json({
success: false,
message: 'Le code a expiré'
});
}
// Check if the code is correct
if (verification.code.toString() === code.toString()) {
verificationCodes.delete(phoneNumber);
res.json({
success: true,
message: 'Code vérifié avec succès'
});
} else {
verification.attempts += 1;
if (verification.attempts >= 3) {
verificationCodes.delete(phoneNumber);
res.status(400).json({
success: false,
message: 'Trop de tentatives. Veuillez demander un nouveau code'
});
} else {
res.status(400).json({
success: false,
message: 'Code incorrect'
});
}
}
});
//------------------------------------ End of SMS Section ------------------------------------
//------------------------------------ Email Section -----------------------------------------
const configEmail = { const config = {
MAILCHIMP_API_KEY: process.env.MAILCHIMP_API_KEY, MAILCHIMP_API_KEY: process.env.MAILCHIMP_API_KEY,
MAILCHIMP_KEY: process.env.MAILCHIMP_KEY, MAILCHIMP_KEY: process.env.MAILCHIMP_KEY,
MAILCHIMP_LIST_ID: process.env.MAILCHIMP_LIST_ID, MAILCHIMP_LIST_ID: process.env.MAILCHIMP_LIST_ID,
@ -465,15 +195,15 @@ const pendingEmails = new Map();
class EmailService { class EmailService {
static async sendTransactionalEmail(to, templateName, subject, templateVariables) { static async sendTransactionalEmail(to, templateName, subject, templateVariables) {
try { try {
const mailchimpClient = mailchimp(configEmail.MAILCHIMP_API_KEY); const mailchimpClient = mailchimp(config.MAILCHIMP_API_KEY);
const message = { const message = {
template_name: templateName, template_name: templateName,
template_content: [], template_content: [],
message: { message: {
global_merge_vars: this.buildVariables(templateVariables), global_merge_vars: this.buildVariables(templateVariables),
from_email: configEmail.FROM_EMAIL, from_email: config.FROM_EMAIL,
from_name: configEmail.FROM_NAME, from_name: config.FROM_NAME,
subject: subject, subject: subject,
to: [ to: [
{ {
@ -502,12 +232,12 @@ class EmailService {
// Add to Mailchimp diffusion list // Add to Mailchimp diffusion list
static async addToMailchimpList(email) { static async addToMailchimpList(email) {
try { try {
const url = `https://us17.api.mailchimp.com/3.0/lists/${configEmail.MAILCHIMP_LIST_ID}/members`; const url = `https://us17.api.mailchimp.com/3.0/lists/${config.MAILCHIMP_LIST_ID}/members`;
const response = await fetch(url, { const response = await fetch(url, {
method: 'POST', method: 'POST',
headers: { headers: {
'Authorization': `apikey ${configEmail.MAILCHIMP_KEY}`, 'Authorization': `apikey ${config.MAILCHIMP_KEY}`,
'Content-Type': 'application/json' 'Content-Type': 'application/json'
}, },
body: JSON.stringify({ body: JSON.stringify({
@ -665,276 +395,11 @@ app.post('/api/subscribe-to-list', validateEmail, async (req, res) => {
} }
}); });
app.post('/api/send_reminder', async (req, res) => {
const { office, customer } = req.body;
try {
const to = customer.contact.email;
const templateVariables = {
office_name: office.name,
last_name: customer.contact.last_name || '',
first_name: customer.contact.first_name || '',
link: `${process.env.APP_HOST}`
};
await EmailService.sendTransactionalEmail(
to,
ETemplates.DOCUMENT_REMINDER,
'Vous avez des documents à déposer pour votre dossier.',
templateVariables
);
res.json({
success: true,
message: 'Email envoyé avec succès'
});
} catch (error) {
console.error(error);
return;
}
});
// Automatic retry system // Automatic retry system
setInterval(() => { setInterval(() => {
EmailService.retryFailedEmails(); EmailService.retryFailedEmails();
}, 60000); // Check every minute }, 60000); // Check every minute
//------------------------------------ End of Email Section ------------------------------------
//------------------------------------ Stripe Section ------------------------------------------
const configStripe = {
STRIPE_SECRET_KEY: process.env.STRIPE_SECRET_KEY,
STRIPE_WEBHOOK_SECRET: process.env.STRIPE_WEBHOOK_SECRET,
APP_HOST: process.env.APP_HOST || 'http://localhost:3000',
};
// Stripe service
class StripeService {
constructor() {
this.client = new Stripe(configStripe.STRIPE_SECRET_KEY);
this.prices = {
STANDARD: {
monthly: process.env.STRIPE_STANDARD_SUBSCRIPTION_PRICE_ID,
yearly: process.env.STRIPE_STANDARD_ANNUAL_SUBSCRIPTION_PRICE_ID
},
UNLIMITED: {
monthly: process.env.STRIPE_UNLIMITED_SUBSCRIPTION_PRICE_ID,
yearly: process.env.STRIPE_UNLIMITED_ANNUAL_SUBSCRIPTION_PRICE_ID
}
};
}
// Only for test
async createTestSubscription() {
try {
const customer = await this.client.customers.create({
email: 'test@example.com',
description: 'Client test',
source: 'tok_visa'
});
const priceId = this.prices.STANDARD.monthly;
const price = await this.client.prices.retrieve(priceId);
const subscription = await this.client.subscriptions.create({
customer: customer.id,
items: [{ price: price.id }],
payment_behavior: 'default_incomplete',
expand: ['latest_invoice.payment_intent']
});
return {
subscriptionId: subscription.id,
customerId: customer.id,
status: subscription.status,
priceId: price.id
};
} catch (error) {
throw error;
}
}
async createCheckoutSession(subscription, frequency) {
const priceId = this.getPriceId(subscription.type, frequency);
return await this.client.checkout.sessions.create({
mode: 'subscription',
payment_method_types: ['card', 'sepa_debit'],
billing_address_collection: 'auto',
line_items: [{
price: priceId,
quantity: subscription.type === 'STANDARD' ? subscription.seats : 1,
}],
success_url: `${configStripe.APP_HOST}/subscription/success`, // Success page (frontend)
cancel_url: `${configStripe.APP_HOST}/subscription/error`, // Error page (frontend)
metadata: {
subscription: JSON.stringify(subscription),
},
allow_promotion_codes: true,
automatic_tax: { enabled: true }
});
}
getPriceId(type, frequency) {
return this.prices[type][frequency];
}
async getSubscription(subscriptionId) {
return await this.client.subscriptions.retrieve(subscriptionId);
}
async createPortalSession(subscriptionId) {
const subscription = await this.getSubscription(subscriptionId);
return await this.client.billingPortal.sessions.create({
customer: subscription.customer,
return_url: `${configStripe.APP_HOST}/subscription/manage`
});
}
}
const stripeService = new StripeService();
// Validation middleware
const validateSubscription = (req, res, next) => {
const { type, seats, frequency } = req.body;
if (!type || !['STANDARD', 'UNLIMITED'].includes(type)) {
return res.status(400).json({
success: false,
message: 'Type d\'abonnement invalide'
});
}
if (type === 'STANDARD' && (!seats || seats < 1)) {
return res.status(400).json({
success: false,
message: 'Nombre de sièges invalide'
});
}
if (!frequency || !['monthly', 'yearly'].includes(frequency)) {
return res.status(400).json({
success: false,
message: 'Fréquence invalide'
});
}
next();
};
// Routes
// Only for test
app.post('/api/test/create-subscription', async (req, res) => {
try {
const result = await stripeService.createTestSubscription();
res.json({
success: true,
data: result
});
} catch (error) {
res.status(500).json({
success: false,
message: 'Erreur lors de la création de l\'abonnement de test',
error: {
message: error.message,
type: error.type,
code: error.code
}
});
}
});
app.post('/api/subscriptions/checkout', validateSubscription, async (req, res) => {
try {
const session = await stripeService.createCheckoutSession(req.body, req.body.frequency);
res.json({ success: true, sessionId: session.id });
} catch (error) {
console.error('Erreur création checkout:', error);
res.status(500).json({
success: false,
message: 'Erreur lors de la création de la session de paiement'
});
}
});
app.get('/api/subscriptions/:id', async (req, res) => {
try {
const subscription = await stripeService.getSubscription(req.params.id);
res.json({ success: true, subscription });
} catch (error) {
res.status(500).json({
success: false,
message: 'Erreur lors de la récupération de l\'abonnement'
});
}
});
app.post('/api/subscriptions/:id/portal', async (req, res) => {
try {
const session = await stripeService.createPortalSession(req.params.id);
res.json({ success: true, url: session.url });
} catch (error) {
res.status(500).json({
success: false,
message: 'Erreur lors de la création de la session du portail'
});
}
});
// Webhook Stripe
app.post('/api/webhooks/stripe', express.raw({ type: 'application/json' }), async (req, res) => {
const sig = req.headers['stripe-signature'];
let event;
try {
event = Stripe.webhooks.constructEvent(req.body, sig, configStripe.STRIPE_WEBHOOK_SECRET);
} catch (err) {
return res.status(400).send(`Webhook Error: ${err.message}`);
}
try {
switch (event.type) {
case 'checkout.session.completed':
const session = event.data.object;
if (session.status === 'complete') {
const subscription = JSON.parse(session.metadata.subscription);
// Stock subscription (create process)
console.log('Nouvel abonnement:', subscription);
}
break;
case 'invoice.payment_succeeded':
const invoice = event.data.object;
if (['subscription_update', 'subscription_cycle'].includes(invoice.billing_reason)) {
const subscription = await stripeService.getSubscription(invoice.subscription);
// Update subscription (update process)
console.log('Mise à jour abonnement:', subscription);
}
break;
case 'customer.subscription.deleted':
const deletedSubscription = event.data.object;
// Delete subscription (update process to delete)
console.log('Suppression abonnement:', deletedSubscription.id);
break;
}
res.json({ received: true });
} catch (error) {
console.error('Erreur webhook:', error);
res.status(500).json({
success: false,
message: 'Erreur lors du traitement du webhook'
});
}
});
//------------------------------------ End of Stripe Section -----------------------------------
app.listen(PORT, () => { app.listen(PORT, () => {
console.log(`Server is running on port ${PORT}`); console.log(`Server is running on port ${PORT}`);
}); });