whatsapp/index.js

222 lines
7.0 KiB
JavaScript
Raw Normal View History

2025-05-06 14:04:11 +00:00
const { default: makeWASocket, useMultiFileAuthState, DisconnectReason, fetchLatestBaileysVersion, proto, generateWAMessageFromContent, generateWAMessageContent } = require('@fizzxydev/baileys-pro');
const express = require('express');
const { Boom } = require('@hapi/boom');
const qrcode = require('qrcode');
const path = require('path');
const fs = require('fs');
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 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,
printQRInTerminal: false
2025-05-06 14:04:11 +00:00
});
2025-05-06 08:51:13 +00:00
2025-05-06 14:04:11 +00:00
sock.ev.on('connection.update', async (update) => {
const { connection, lastDisconnect, qr } = update;
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
}
if (connection === 'close') {
2025-05-06 14:04:11 +00:00
const shouldReconnect = lastDisconnect?.error?.output?.statusCode !== DisconnectReason.loggedOut;
if (shouldReconnect) {
console.log('🔁 Reconnexion...');
initBaileys();
} else {
console.log('❌ Déconnecté.');
}
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 10:12:55 +00:00
2025-05-06 14:04:11 +00:00
initBaileys();
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 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;
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
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 12:28:20 +00:00
message: {
interactiveMessage: proto.Message.InteractiveMessage.create({
2025-05-06 14:04:11 +00:00
body: { text: "Bienvenue sur notre service !" },
footer: { text: "Choisis une action ci-dessous" },
header: {
2025-05-06 12:28:20 +00:00
title: "Menu principal",
hasMediaAttachment: false
2025-05-06 14:04:11 +00:00
},
nativeFlowMessage: {
2025-05-06 12:28:20 +00:00
buttons: [
{
name: "cta_reply",
buttonParamsJson: JSON.stringify({
display_text: "📩 Contacter support",
id: "support_action"
})
},
{
name: "cta_url",
buttonParamsJson: JSON.stringify({
display_text: "🌐 Voir notre site",
url: "https://canguidev.fr",
merchant_url: "https://canguidev.fr"
})
},
{
name: "cta_call",
buttonParamsJson: JSON.stringify({
display_text: "📞 Appeler le support",
id: "+33612345678"
})
}
]
2025-05-06 14:04:11 +00:00
}
2025-05-06 12:28:20 +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 14:04:11 +00:00
const msg = generateWAMessageFromContent(`${phone}@s.whatsapp.net`, content, {});
await sock.relayMessage(`${phone}@s.whatsapp.net`, 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 12:17:18 +00:00
app.post('/sendInteractiveImage', async (req, res) => {
2025-05-06 14:04:11 +00:00
const { phone, caption, title, subtitle, footer } = req.body;
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
2025-05-06 12:17:18 +00:00
try {
2025-05-06 14:15:39 +00:00
const { prepareWAMessageMedia } = require('@fizzxydev/baileys-pro');
2025-05-06 14:04:11 +00:00
const imagePath = path.join(__dirname, 'public', 'logo-merlo-cs-FR.jpg');
2025-05-06 14:10:28 +00:00
const imageBuffer = fs.readFileSync(imagePath);
2025-05-06 14:15:39 +00:00
// Préparation du média avec la méthode officielle
const preparedImage = await prepareWAMessageMedia(
{ image: imageBuffer },
{
upload: sock.aws.uploadStream, // Méthode correcte d'upload
mediaType: 'image',
options: { contentType: 'image/jpeg' }
}
);
2025-05-06 14:13:06 +00:00
// Construction du message interactif
2025-05-06 14:10:28 +00:00
const message = {
interactive: {
type: 'button',
header: {
type: 'IMAGE',
2025-05-06 14:15:39 +00:00
image: preparedImage.image, // Utilisation du média préparé
2025-05-06 14:10:28 +00:00
title: title || 'Titre par défaut',
subtitle: subtitle || 'Sous-titre',
},
body: {
text: caption || 'Description par défaut'
},
footer: {
text: footer || 'Pied de page'
2025-05-06 14:04:11 +00:00
},
2025-05-06 14:10:28 +00:00
action: {
buttons: [
{
type: 'url',
title: 'Proposition',
url: 'https://merlo-ch.com/uploads/proposition/f_p_250505_0000136_00008_EB00001909.pdf'
},
{
type: 'url',
title: 'Spec machine',
url: 'https://merlo-ch.com/uploads/proposition/d_p_250505_0000136_00008_EB00001909.pdf'
}
]
2025-05-06 12:17:18 +00:00
}
2025-05-06 14:10:28 +00:00
}
};
await sock.sendMessage(`${phone}@s.whatsapp.net`, message);
2025-05-06 14:04:11 +00:00
res.json({ success: true });
} catch (e) {
console.error('❌ Erreur interactive image :', e);
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:10:28 +00:00
// app.post('/sendInteractiveImage', 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 12:17:18 +00:00
2025-05-06 14:04:11 +00:00
app.listen(3001, () => console.log('🚀 Serveur Baileys démarré sur http://localhost:3001'));