2025-05-06 14:04:11 +00:00
|
|
|
|
const express = require('express');
|
|
|
|
|
|
const qrcode = require('qrcode');
|
|
|
|
|
|
const path = require('path');
|
|
|
|
|
|
const fs = require('fs');
|
2025-05-06 20:25:52 +00:00
|
|
|
|
const NodeCache = require('node-cache');
|
2025-05-06 19:25:43 +00:00
|
|
|
|
const {
|
2025-05-06 20:21:45 +00:00
|
|
|
|
default: makeWASocket,
|
2025-05-06 19:25:43 +00:00
|
|
|
|
useMultiFileAuthState,
|
|
|
|
|
|
DisconnectReason,
|
|
|
|
|
|
fetchLatestBaileysVersion,
|
2025-05-06 20:25:52 +00:00
|
|
|
|
proto,
|
|
|
|
|
|
generateWAMessageFromContent
|
|
|
|
|
|
} = require('@fizzxydev/baileys-pro');
|
2025-05-06 10:12:55 +00:00
|
|
|
|
|
2025-05-06 14:04:11 +00:00
|
|
|
|
const app = express();
|
|
|
|
|
|
app.use(express.json());
|
|
|
|
|
|
app.use(express.static('public'));
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 14:04:11 +00:00
|
|
|
|
let sock;
|
|
|
|
|
|
let qrData = null;
|
|
|
|
|
|
let isConnected = false;
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 20:25:52 +00:00
|
|
|
|
// 💾 Group Metadata Cache (5 minutes)
|
|
|
|
|
|
const groupCache = new NodeCache({ stdTTL: 300, useClones: false });
|
|
|
|
|
|
|
|
|
|
|
|
// Simule une récupération de message depuis un store pour getMessage
|
|
|
|
|
|
async function getMessageFromStore(key) {
|
|
|
|
|
|
// À adapter selon ta logique réelle
|
|
|
|
|
|
return { conversation: "Message temporaire pour retry" };
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-06 10:12:55 +00:00
|
|
|
|
const initBaileys = async () => {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
const { version } = await fetchLatestBaileysVersion();
|
|
|
|
|
|
const { state, saveCreds } = await useMultiFileAuthState('auth');
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 10:12:55 +00:00
|
|
|
|
sock = makeWASocket({
|
|
|
|
|
|
version,
|
|
|
|
|
|
auth: state,
|
2025-05-06 20:25:52 +00:00
|
|
|
|
markOnlineOnConnect: false,
|
|
|
|
|
|
getMessage: async (key) => await getMessageFromStore(key),
|
|
|
|
|
|
cachedGroupMetadata: async (jid) => groupCache.get(jid)
|
2025-05-06 14:04:11 +00:00
|
|
|
|
});
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 19:25:43 +00:00
|
|
|
|
sock.ev.on('connection.update', async ({ connection, lastDisconnect, qr }) => {
|
2025-05-06 10:12:55 +00:00
|
|
|
|
if (qr) {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
qrData = await qrcode.toDataURL(qr);
|
|
|
|
|
|
isConnected = false;
|
2025-05-06 10:12:55 +00:00
|
|
|
|
}
|
2025-05-06 19:25:43 +00:00
|
|
|
|
|
2025-05-06 10:12:55 +00:00
|
|
|
|
if (connection === 'close') {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
const shouldReconnect = lastDisconnect?.error?.output?.statusCode !== DisconnectReason.loggedOut;
|
2025-05-06 20:25:52 +00:00
|
|
|
|
console.log(shouldReconnect ? '🔁 Reconnexion...' : '❌ Déconnecté.');
|
|
|
|
|
|
if (shouldReconnect) initBaileys();
|
2025-05-06 10:12:55 +00:00
|
|
|
|
} else if (connection === 'open') {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
console.log('✅ Connecté à WhatsApp');
|
|
|
|
|
|
isConnected = true;
|
2025-05-06 10:12:55 +00:00
|
|
|
|
}
|
2025-05-06 14:04:11 +00:00
|
|
|
|
});
|
2025-05-06 13:54:09 +00:00
|
|
|
|
|
2025-05-06 14:04:11 +00:00
|
|
|
|
sock.ev.on('creds.update', saveCreds);
|
2025-05-06 19:25:43 +00:00
|
|
|
|
|
2025-05-06 20:25:52 +00:00
|
|
|
|
// 📌 Caching des groupes
|
|
|
|
|
|
sock.ev.on('groups.update', async ([event]) => {
|
|
|
|
|
|
const metadata = await sock.groupMetadata(event.id);
|
|
|
|
|
|
groupCache.set(event.id, metadata);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
sock.ev.on('group-participants.update', async (event) => {
|
|
|
|
|
|
const metadata = await sock.groupMetadata(event.id);
|
|
|
|
|
|
groupCache.set(event.id, metadata);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
// (Facultatif) Gestion des messages reçus
|
|
|
|
|
|
sock.ev.on('messages.upsert', async ({ messages }) => {
|
|
|
|
|
|
const msg = messages[0];
|
2025-05-07 05:19:18 +00:00
|
|
|
|
|
2025-05-06 20:25:52 +00:00
|
|
|
|
if (!msg.key.fromMe && msg.message?.conversation) {
|
|
|
|
|
|
console.log('💬 Message reçu de', msg.key.remoteJid, ':', msg.message.conversation);
|
|
|
|
|
|
}
|
2025-05-07 05:19:18 +00:00
|
|
|
|
if (!msg.message || !msg.key.fromMe) {
|
|
|
|
|
|
const jid = msg.key.remoteJid;
|
|
|
|
|
|
const buttonId = msg.message.buttonsResponseMessage?.selectedButtonId;
|
|
|
|
|
|
|
|
|
|
|
|
if (buttonId === 'doc_1') {
|
|
|
|
|
|
await sock.sendMessage(jid, {
|
|
|
|
|
|
document: {
|
2025-05-07 05:34:09 +00:00
|
|
|
|
url: 'https://merlo-ch.com/uploads/proposition/f_p_250505_0000136_00008_EB00001909.pdf',
|
2025-05-07 05:19:18 +00:00
|
|
|
|
},
|
|
|
|
|
|
mimetype: 'application/pdf',
|
|
|
|
|
|
fileName: 'Document_1.pdf',
|
|
|
|
|
|
caption: 'Voici votre *Document 1* 📄',
|
|
|
|
|
|
footer: '© Fizzxy Dev',
|
|
|
|
|
|
});
|
|
|
|
|
|
} else if (buttonId === 'doc_2') {
|
|
|
|
|
|
await sock.sendMessage(jid, {
|
|
|
|
|
|
document: {
|
2025-05-07 05:34:09 +00:00
|
|
|
|
url: 'https://merlo-ch.com/uploads/proposition/d_p_250505_0000136_00008_EB00001909.pdf',
|
2025-05-07 05:19:18 +00:00
|
|
|
|
},
|
|
|
|
|
|
mimetype: 'application/pdf',
|
|
|
|
|
|
fileName: 'Document_2.pdf',
|
|
|
|
|
|
caption: 'Voici votre *Document 2* 📄',
|
|
|
|
|
|
footer: '© Fizzxy Dev',
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
2025-05-07 05:16:23 +00:00
|
|
|
|
}
|
2025-05-06 20:25:52 +00:00
|
|
|
|
});
|
2025-05-06 14:04:11 +00:00
|
|
|
|
};
|
2025-05-06 10:12:55 +00:00
|
|
|
|
|
2025-05-06 14:04:11 +00:00
|
|
|
|
initBaileys();
|
2025-05-06 20:25:52 +00:00
|
|
|
|
|
2025-05-06 14:31:16 +00:00
|
|
|
|
app.use('/static', express.static(path.join(__dirname, 'public')));
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 09:54:23 +00:00
|
|
|
|
app.get('/login', (req, res) => {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
res.sendFile(path.join(__dirname, 'public', 'login.html'));
|
|
|
|
|
|
});
|
2025-05-06 10:12:55 +00:00
|
|
|
|
|
2025-05-06 09:54:23 +00:00
|
|
|
|
app.get('/api/qrcode', (req, res) => {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
res.json({ qr: qrData, connected: isConnected });
|
|
|
|
|
|
});
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 09:38:23 +00:00
|
|
|
|
app.post('/sendText', async (req, res) => {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
const { phone, message } = req.body;
|
|
|
|
|
|
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
|
2025-05-06 09:38:23 +00:00
|
|
|
|
try {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
await sock.sendMessage(`${phone}@s.whatsapp.net`, { text: message });
|
|
|
|
|
|
res.json({ success: true });
|
2025-05-06 09:54:23 +00:00
|
|
|
|
} catch (e) {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
res.status(500).json({ error: e.message });
|
2025-05-06 09:38:23 +00:00
|
|
|
|
}
|
2025-05-06 14:04:11 +00:00
|
|
|
|
});
|
2025-05-06 20:25:52 +00:00
|
|
|
|
app.post('/testButtons', async (req, res) => {
|
|
|
|
|
|
const { phone } = req.body;
|
2025-05-06 09:43:48 +00:00
|
|
|
|
|
2025-05-06 20:25:52 +00:00
|
|
|
|
if (!sock || !isConnected) {
|
|
|
|
|
|
return res.status(400).json({ error: 'Non connecté' });
|
2025-05-06 11:53:28 +00:00
|
|
|
|
}
|
2025-05-06 19:14:46 +00:00
|
|
|
|
|
2025-05-06 20:25:52 +00:00
|
|
|
|
const jid = `${phone}@s.whatsapp.net`;
|
2025-05-06 20:21:45 +00:00
|
|
|
|
|
|
|
|
|
|
try {
|
2025-05-06 20:25:52 +00:00
|
|
|
|
await sock.sendMessage(jid, {
|
|
|
|
|
|
image: { url : "https://wa.canguidev.fr/static/logo-merlo-cs-FR.jpg" }, // Can buffer
|
|
|
|
|
|
text: "Hello Wolrd !;",
|
|
|
|
|
|
caption: "Description Of Messages", //Additional information
|
|
|
|
|
|
footer: "© Fizzxy Dev",
|
|
|
|
|
|
media:true,
|
|
|
|
|
|
image: {
|
|
|
|
|
|
url: 'https://wa.canguidev.fr/static/logo-merlo-cs-FR.jpg'
|
|
|
|
|
|
},
|
|
|
|
|
|
buttons: [
|
|
|
|
|
|
{
|
|
|
|
|
|
buttonId: '.tes',
|
|
|
|
|
|
buttonText: {
|
|
|
|
|
|
displayText: 'TESTING BOT'
|
|
|
|
|
|
},
|
|
|
|
|
|
type: 1
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
buttonId: ' ',
|
|
|
|
|
|
buttonText: {
|
|
|
|
|
|
displayText: 'PRIVATE SCRIPT'
|
|
|
|
|
|
},
|
|
|
|
|
|
type: 1
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
buttonId: 'action',
|
|
|
|
|
|
buttonText: {
|
|
|
|
|
|
displayText: 'ini pesan interactiveMeta'
|
|
|
|
|
|
},
|
|
|
|
|
|
type: 4,
|
|
|
|
|
|
nativeFlowInfo: {
|
|
|
|
|
|
name: 'single_select',
|
|
|
|
|
|
paramsJson: JSON.stringify({
|
|
|
|
|
|
title: 'message',
|
|
|
|
|
|
sections: [
|
|
|
|
|
|
{
|
|
|
|
|
|
title: 'FizzxyDev - 2025',
|
|
|
|
|
|
highlight_label: '😜',
|
|
|
|
|
|
rows: [
|
|
|
|
|
|
{
|
|
|
|
|
|
header: 'HEADER',
|
|
|
|
|
|
title: 'TITLE',
|
|
|
|
|
|
description: 'DESCRIPTION',
|
|
|
|
|
|
id: 'YOUR ID 1'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
header: 'HEADER',
|
|
|
|
|
|
title: 'TITLE',
|
|
|
|
|
|
description: 'DESCRIPTION',
|
|
|
|
|
|
id: 'YOUR ID 2'
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
],
|
|
|
|
|
|
headerType: 1,
|
|
|
|
|
|
viewOnce: true
|
|
|
|
|
|
}, {
|
|
|
|
|
|
quoted: null // ou tu peux injecter un msg pour reply ici
|
2025-05-06 20:21:45 +00:00
|
|
|
|
});
|
2025-05-06 20:25:52 +00:00
|
|
|
|
|
2025-05-06 20:21:45 +00:00
|
|
|
|
res.json({ success: true });
|
|
|
|
|
|
} catch (e) {
|
2025-05-06 20:25:52 +00:00
|
|
|
|
console.error('❌ Erreur testButtons :', e);
|
2025-05-06 20:21:45 +00:00
|
|
|
|
res.status(500).json({ error: e.message });
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2025-05-06 09:43:48 +00:00
|
|
|
|
|
2025-05-06 09:54:23 +00:00
|
|
|
|
app.post('/sendButtons', async (req, res) => {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
const { phone } = req.body;
|
2025-05-06 20:25:52 +00:00
|
|
|
|
|
|
|
|
|
|
if (!sock || !isConnected) {
|
|
|
|
|
|
return res.status(400).json({ error: 'Non connecté' });
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const BASE_URL = process.env.BASE_URL || 'https://wa.canguidev.fr';
|
|
|
|
|
|
const imageUrl = `${BASE_URL}/static/logo-merlo-cs-FR.jpg`;
|
|
|
|
|
|
const jid = `${phone}@s.whatsapp.net`;
|
2025-05-06 12:00:21 +00:00
|
|
|
|
|
2025-05-06 11:53:28 +00:00
|
|
|
|
try {
|
2025-05-06 12:00:21 +00:00
|
|
|
|
const content = {
|
2025-05-06 20:21:45 +00:00
|
|
|
|
viewOnceMessage: {
|
|
|
|
|
|
message: {
|
|
|
|
|
|
messageContextInfo: {
|
|
|
|
|
|
deviceListMetadata: {},
|
|
|
|
|
|
deviceListMetadataVersion: 2
|
2025-05-06 14:04:11 +00:00
|
|
|
|
},
|
2025-05-06 20:21:45 +00:00
|
|
|
|
interactiveMessage: proto.Message.InteractiveMessage.create({
|
2025-05-06 20:25:52 +00:00
|
|
|
|
body: { text: "Bienvenue sur notre service !" },
|
|
|
|
|
|
footer: { text: "Choisis une action ci-dessous" },
|
|
|
|
|
|
header: {
|
|
|
|
|
|
hasMediaAttachment: true,
|
|
|
|
|
|
imageMessage: { url: imageUrl }
|
|
|
|
|
|
},
|
2025-05-06 20:21:45 +00:00
|
|
|
|
nativeFlowMessage: {
|
|
|
|
|
|
buttons: [
|
|
|
|
|
|
{
|
2025-05-06 20:25:52 +00:00
|
|
|
|
name: "cta_reply",
|
|
|
|
|
|
buttonParamsJson: JSON.stringify({
|
|
|
|
|
|
display_text: "📩 Contacter support",
|
|
|
|
|
|
id: "support_action"
|
|
|
|
|
|
})
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
name: "cta_url",
|
2025-05-06 20:21:45 +00:00
|
|
|
|
buttonParamsJson: JSON.stringify({
|
2025-05-06 20:25:52 +00:00
|
|
|
|
display_text: "🌐 Voir notre site",
|
|
|
|
|
|
url: "https://canguidev.fr"
|
2025-05-06 20:21:45 +00:00
|
|
|
|
})
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
2025-05-06 20:25:52 +00:00
|
|
|
|
name: "cta_call",
|
2025-05-06 20:21:45 +00:00
|
|
|
|
buttonParamsJson: JSON.stringify({
|
2025-05-06 20:25:52 +00:00
|
|
|
|
display_text: "📞 Appeler le support",
|
|
|
|
|
|
id: "+33612345678"
|
2025-05-06 20:21:45 +00:00
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
2025-05-06 12:00:21 +00:00
|
|
|
|
}
|
2025-05-06 14:04:11 +00:00
|
|
|
|
};
|
2025-05-06 12:08:18 +00:00
|
|
|
|
|
2025-05-06 20:16:28 +00:00
|
|
|
|
const msg = generateWAMessageFromContent(jid, content, {});
|
|
|
|
|
|
await sock.relayMessage(jid, msg.message, { messageId: msg.key.id });
|
2025-05-06 11:53:28 +00:00
|
|
|
|
|
2025-05-06 14:04:11 +00:00
|
|
|
|
res.json({ success: true });
|
2025-05-06 11:53:28 +00:00
|
|
|
|
} catch (e) {
|
2025-05-06 14:04:11 +00:00
|
|
|
|
console.error('❌ Erreur bouton actif :', e);
|
|
|
|
|
|
res.status(500).json({ error: e.message });
|
2025-05-06 11:53:28 +00:00
|
|
|
|
}
|
2025-05-06 14:04:11 +00:00
|
|
|
|
});
|
2025-05-06 20:25:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
2025-05-06 14:52:28 +00:00
|
|
|
|
// Votre route POST
|
2025-05-06 12:17:18 +00:00
|
|
|
|
app.post('/sendInteractiveImage', async (req, res) => {
|
2025-05-06 15:21:46 +00:00
|
|
|
|
const { phone, caption, title, subtitle, footer } = req.body;
|
2025-05-06 14:17:53 +00:00
|
|
|
|
if (!sock || !isConnected) {
|
2025-05-06 14:52:28 +00:00
|
|
|
|
return res.status(400).json({ error: 'Non connecté à WhatsApp' });
|
2025-05-06 14:17:53 +00:00
|
|
|
|
}
|
2025-05-06 12:17:18 +00:00
|
|
|
|
|
|
|
|
|
|
try {
|
2025-05-06 14:57:11 +00:00
|
|
|
|
const BASE_URL = process.env.BASE_URL || 'https://wa.canguidev.fr';
|
2025-05-06 14:49:43 +00:00
|
|
|
|
const imageUrl = `${BASE_URL}/static/logo-merlo-cs-FR.jpg`;
|
2025-05-06 14:39:57 +00:00
|
|
|
|
|
2025-05-06 15:21:46 +00:00
|
|
|
|
// Construire le Header correct
|
2025-05-06 20:25:52 +00:00
|
|
|
|
const header = proto.Message.InteractiveMessage.create({
|
|
|
|
|
|
// ici on injecte l'image dans le header sans besoin de type explicite
|
|
|
|
|
|
header: proto.Message.InteractiveMessage.Header.create({
|
|
|
|
|
|
title: 'Igna',
|
|
|
|
|
|
subtitle: 'test'
|
|
|
|
|
|
}),
|
|
|
|
|
|
})
|
2025-05-06 15:21:46 +00:00
|
|
|
|
// Body et Footer
|
|
|
|
|
|
const body = proto.Message.InteractiveMessage.Body.create({
|
|
|
|
|
|
text: caption || 'Description par défaut'
|
|
|
|
|
|
});
|
|
|
|
|
|
const foot = proto.Message.InteractiveMessage.Footer.create({
|
|
|
|
|
|
text: footer || 'Pied de page'
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
// Vos boutons URL
|
|
|
|
|
|
const nativeFlow = proto.Message.InteractiveMessage.NativeFlowMessage.create({
|
|
|
|
|
|
buttons: [
|
|
|
|
|
|
{
|
|
|
|
|
|
name: 'cta_url',
|
|
|
|
|
|
buttonParamsJson: JSON.stringify({
|
|
|
|
|
|
display_text: '📄 Voir proposition',
|
|
|
|
|
|
url: 'https://merlo-ch.com/uploads/proposition/f_p_250505_0000136_00008_EB00001909.pdf'
|
|
|
|
|
|
})
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
name: 'cta_url',
|
|
|
|
|
|
buttonParamsJson: JSON.stringify({
|
|
|
|
|
|
display_text: '🔧 Spécifications',
|
|
|
|
|
|
url: 'https://merlo-ch.com/uploads/proposition/d_p_250505_0000136_00008_EB00001909.pdf'
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
// Construire l'InteractiveMessage complet
|
|
|
|
|
|
const interactiveMsg = proto.Message.InteractiveMessage.create({
|
|
|
|
|
|
header,
|
|
|
|
|
|
body,
|
|
|
|
|
|
footer: foot,
|
|
|
|
|
|
nativeFlowMessage: nativeFlow
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
// Envelopper (ici dans viewOnceMessage, comme dans votre exemple)
|
|
|
|
|
|
const raw = {
|
2025-05-06 14:52:28 +00:00
|
|
|
|
viewOnceMessage: {
|
|
|
|
|
|
message: {
|
|
|
|
|
|
messageContextInfo: {
|
|
|
|
|
|
deviceListMetadata: {},
|
|
|
|
|
|
deviceListMetadataVersion: 2
|
2025-05-06 14:49:43 +00:00
|
|
|
|
},
|
2025-05-06 15:21:46 +00:00
|
|
|
|
interactiveMessage: interactiveMsg
|
2025-05-06 14:49:43 +00:00
|
|
|
|
}
|
2025-05-06 14:37:18 +00:00
|
|
|
|
}
|
2025-05-06 14:52:28 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
2025-05-06 15:21:46 +00:00
|
|
|
|
// Générer et relayer
|
2025-05-06 14:52:28 +00:00
|
|
|
|
const jid = `${phone}@s.whatsapp.net`;
|
2025-05-06 15:21:46 +00:00
|
|
|
|
const msg = generateWAMessageFromContent(jid, raw, {});
|
2025-05-06 14:52:28 +00:00
|
|
|
|
await sock.relayMessage(jid, msg.message, { messageId: msg.key.id });
|
2025-05-06 14:49:43 +00:00
|
|
|
|
|
|
|
|
|
|
return res.json({ success: true });
|
2025-05-06 14:04:11 +00:00
|
|
|
|
} catch (e) {
|
2025-05-06 14:52:28 +00:00
|
|
|
|
console.error('❌ Erreur /sendInteractiveImage :', e);
|
2025-05-06 14:49:43 +00:00
|
|
|
|
return res.status(500).json({ error: e.message });
|
2025-05-06 12:17:18 +00:00
|
|
|
|
}
|
2025-05-06 14:04:11 +00:00
|
|
|
|
});
|
2025-05-06 14:52:28 +00:00
|
|
|
|
|
2025-05-07 06:11:04 +00:00
|
|
|
|
app.post('/sendInteractiveImage2', async (req, res) => {
|
|
|
|
|
|
const { phone, caption, title, subtitle, footer } = req.body;
|
|
|
|
|
|
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
const imagePath = path.join(__dirname, 'public', 'logo-merlo-cs-FR.jpg');
|
|
|
|
|
|
const resizedBuffer = fs.readFileSync(imagePath);
|
|
|
|
|
|
|
|
|
|
|
|
await sock.sendMessage(`${phone}@s.whatsapp.net`, {
|
|
|
|
|
|
image: resizedBuffer,
|
|
|
|
|
|
caption: caption || 'Description par défaut',
|
|
|
|
|
|
title: title || 'Titre par défaut',
|
|
|
|
|
|
subtitle: subtitle || 'Sous-titre',
|
|
|
|
|
|
footer: footer || 'Pied de page',
|
|
|
|
|
|
media: true,
|
|
|
|
|
|
interactiveButtons: [
|
|
|
|
|
|
{
|
|
|
|
|
|
name: 'cta_url',
|
|
|
|
|
|
buttonParamsJson: JSON.stringify({
|
|
|
|
|
|
display_text: 'Proposition',
|
|
|
|
|
|
url: 'https://merlo-ch.com/uploads/proposition/f_p_250505_0000136_00008_EB00001909.pdf'
|
|
|
|
|
|
})
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
name: 'cta_url',
|
|
|
|
|
|
buttonParamsJson: JSON.stringify({
|
|
|
|
|
|
display_text: 'Spec machine',
|
|
|
|
|
|
url: 'https://merlo-ch.com/uploads/proposition/d_p_250505_0000136_00008_EB00001909.pdf'
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
});
|
|
|
|
|
|
res.json({ success: true });
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error('❌ Erreur interactive image :', e);
|
|
|
|
|
|
res.status(500).json({ error: e.message });
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2025-05-06 15:28:38 +00:00
|
|
|
|
app.post('/sendProductMessage', async (req, res) => {
|
2025-05-06 15:35:50 +00:00
|
|
|
|
// On vérifie sock et isConnected, pas client
|
|
|
|
|
|
if (!sock || !isConnected) {
|
2025-05-06 15:28:38 +00:00
|
|
|
|
return res.status(400).json({ error: 'Non connecté à WhatsApp' });
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const {
|
|
|
|
|
|
phone,
|
|
|
|
|
|
productImageUrl,
|
|
|
|
|
|
productImageCount,
|
|
|
|
|
|
productTitle,
|
|
|
|
|
|
productDescription,
|
|
|
|
|
|
priceAmount1000,
|
|
|
|
|
|
currencyCode,
|
|
|
|
|
|
retailerId,
|
|
|
|
|
|
productUrl,
|
|
|
|
|
|
businessOwnerJid,
|
|
|
|
|
|
caption,
|
|
|
|
|
|
messageTitle,
|
|
|
|
|
|
footer,
|
|
|
|
|
|
interactiveButtons = [],
|
|
|
|
|
|
quoted
|
|
|
|
|
|
} = req.body;
|
2025-05-06 12:17:18 +00:00
|
|
|
|
|
2025-05-06 15:28:38 +00:00
|
|
|
|
try {
|
|
|
|
|
|
const jid = `${phone}@s.whatsapp.net`;
|
|
|
|
|
|
|
2025-05-06 15:35:50 +00:00
|
|
|
|
// Build product payload exactly comme avant
|
2025-05-06 15:28:38 +00:00
|
|
|
|
const productPayload = {
|
2025-05-06 15:35:50 +00:00
|
|
|
|
productImage: { url: productImageUrl },
|
2025-05-06 15:28:38 +00:00
|
|
|
|
productImageCount,
|
2025-05-06 15:35:50 +00:00
|
|
|
|
title: productTitle, // ici productTitle
|
2025-05-06 15:34:25 +00:00
|
|
|
|
description: productDescription,
|
2025-05-06 15:35:50 +00:00
|
|
|
|
priceAmount1000: priceAmount1000 * 1000, // ajustez si besoin
|
2025-05-06 15:28:38 +00:00
|
|
|
|
currencyCode,
|
|
|
|
|
|
retailerId,
|
2025-05-06 15:34:25 +00:00
|
|
|
|
url: productUrl
|
2025-05-06 15:28:38 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
2025-05-06 15:35:50 +00:00
|
|
|
|
// Transformer les boutons
|
2025-05-06 15:28:38 +00:00
|
|
|
|
const buttons = interactiveButtons.map(btn => {
|
|
|
|
|
|
const params = {};
|
2025-05-06 15:34:25 +00:00
|
|
|
|
if (btn.id) params.id = btn.id;
|
|
|
|
|
|
if (btn.url) params.url = btn.url;
|
2025-05-06 15:28:38 +00:00
|
|
|
|
if (btn.display_text) params.display_text = btn.display_text;
|
|
|
|
|
|
return {
|
2025-05-06 15:35:50 +00:00
|
|
|
|
name: btn.name,
|
|
|
|
|
|
buttonParamsJson: JSON.stringify(params)
|
2025-05-06 15:28:38 +00:00
|
|
|
|
};
|
|
|
|
|
|
});
|
|
|
|
|
|
|
2025-05-06 15:35:50 +00:00
|
|
|
|
// **On utilise sock.sendMessage** et non client.sendMessage
|
|
|
|
|
|
await sock.sendMessage(
|
2025-05-06 15:28:38 +00:00
|
|
|
|
jid,
|
|
|
|
|
|
{
|
2025-05-06 15:35:50 +00:00
|
|
|
|
product: productPayload,
|
2025-05-06 15:28:38 +00:00
|
|
|
|
businessOwnerJid,
|
|
|
|
|
|
caption,
|
2025-05-06 15:35:50 +00:00
|
|
|
|
title: messageTitle,
|
2025-05-06 15:28:38 +00:00
|
|
|
|
footer,
|
2025-05-06 15:35:50 +00:00
|
|
|
|
media: true,
|
|
|
|
|
|
interactiveButtons: buttons
|
2025-05-06 15:28:38 +00:00
|
|
|
|
},
|
|
|
|
|
|
quoted ? { quoted } : {}
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
return res.json({ success: true });
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error('❌ Erreur /sendProductMessage :', e);
|
|
|
|
|
|
return res.status(500).json({ error: e.message });
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2025-05-07 07:11:49 +00:00
|
|
|
|
app.post('/testViewOnce', async (req, res) => {
|
|
|
|
|
|
const { phone } = req.body;
|
|
|
|
|
|
|
|
|
|
|
|
if (!sock || !isConnected) {
|
|
|
|
|
|
return res.status(400).json({ error: 'Non connecté' });
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const jid = `${phone}@s.whatsapp.net`;
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
await sock.sendMessage(jid, {
|
|
|
|
|
|
image: {
|
|
|
|
|
|
url: 'https://wa.canguidev.fr/static/logo-merlo-cs-FR.jpg' // Ton image publique
|
|
|
|
|
|
},
|
|
|
|
|
|
viewOnce: true,
|
|
|
|
|
|
caption: 'Voici une image à affichage unique 📷'
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
res.json({ success: true });
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error('❌ Erreur testViewOnce :', e);
|
|
|
|
|
|
res.status(500).json({ error: e.message });
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2025-05-07 07:19:48 +00:00
|
|
|
|
app.post('/testHeaderImage', async (req, res) => {
|
|
|
|
|
|
const { phone } = req.body;
|
|
|
|
|
|
|
|
|
|
|
|
if (!sock || !isConnected) {
|
|
|
|
|
|
return res.status(400).json({ error: 'Non connecté' });
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const jid = `${phone}@s.whatsapp.net`;
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
2025-05-07 07:28:35 +00:00
|
|
|
|
const message = {
|
|
|
|
|
|
viewOnce: true,
|
|
|
|
|
|
interactiveMessage: proto.Message.InteractiveMessage.create({
|
|
|
|
|
|
body: { text: "Voici un message interactif avec un en-tête image." },
|
|
|
|
|
|
footer: { text: "Fizzxy Dev © 2025" },
|
|
|
|
|
|
header: {
|
|
|
|
|
|
hasMediaAttachment: true,
|
|
|
|
|
|
imageMessage: {
|
|
|
|
|
|
url: 'https://wa.canguidev.fr/static/logo-merlo-cs-FR.jpg',
|
|
|
|
|
|
mimetype: 'image/jpeg'
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
nativeFlowMessage: {
|
|
|
|
|
|
buttons: [
|
|
|
|
|
|
{
|
|
|
|
|
|
name: 'cta_reply',
|
|
|
|
|
|
buttonParamsJson: JSON.stringify({
|
|
|
|
|
|
display_text: 'Répondre',
|
|
|
|
|
|
id: 'reply_button'
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const msg = await generateWAMessageFromContent(jid, {
|
|
|
|
|
|
viewOnceMessage: { message: message }
|
|
|
|
|
|
}, {});
|
|
|
|
|
|
|
|
|
|
|
|
await sock.relayMessage(jid, msg.message, { messageId: msg.key.id });
|
2025-05-07 07:19:48 +00:00
|
|
|
|
|
|
|
|
|
|
res.json({ success: true });
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error('❌ Erreur testHeaderImage :', e);
|
|
|
|
|
|
res.status(500).json({ error: e.message });
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2025-05-06 15:34:25 +00:00
|
|
|
|
|
2025-05-06 15:35:50 +00:00
|
|
|
|
|
2025-05-06 15:34:25 +00:00
|
|
|
|
// 5) 404 et gestion des erreurs
|
|
|
|
|
|
app.use((req, res) => res.status(404).json({ error: 'Ressource introuvable' }));
|
|
|
|
|
|
app.use((err, req, res, next) => {
|
|
|
|
|
|
console.error('Middleware d’erreur :', err);
|
|
|
|
|
|
res.status(500).json({ error: err.message });
|
|
|
|
|
|
});
|
2025-05-06 14:04:11 +00:00
|
|
|
|
app.listen(3001, () => console.log('🚀 Serveur Baileys démarré sur http://localhost:3001'));
|