whatsapp/index.js
2025-05-06 14:00:21 +02:00

245 lines
7.4 KiB
JavaScript

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');
const app = express();
app.use(express.json());
app.use(express.static('public'));
let sock;
let qrData = null;
let isConnected = false;
const initBaileys = async () => {
const { version } = await fetchLatestBaileysVersion();
const { state, saveCreds } = await useMultiFileAuthState('auth');
sock = makeWASocket({
version,
auth: state,
printQRInTerminal: false
});
sock.ev.on('connection.update', async (update) => {
const { connection, lastDisconnect, qr } = update;
if (qr) {
qrData = await qrcode.toDataURL(qr);
isConnected = false;
}
if (connection === 'close') {
const shouldReconnect = lastDisconnect?.error?.output?.statusCode !== DisconnectReason.loggedOut;
if (shouldReconnect) {
console.log('🔁 Reconnexion...');
initBaileys();
} else {
console.log('❌ Déconnecté.');
}
} else if (connection === 'open') {
console.log('✅ Connecté à WhatsApp');
isConnected = true;
}
});
sock.ev.on('creds.update', saveCreds);
};
async function generateMediaMessage(sock, type, url) {
const generated = await generateWAMessageContent(
{ [type]: { url } },
{ upload: sock.waUploadToServer }
);
return generated[`${type}Message`];
}
async function sendInteractive(sock, jid, imageUrl) {
const imageMsg = await generateMediaMessage(sock, 'image', imageUrl);
const content = {
viewOnceMessage: {
message: {
messageContextInfo: {
deviceListMetadata: {},
deviceListMetadataVersion: 2,
},
interactiveMessage: proto.Message.InteractiveMessage.create({
body: { text: 'Quel produit vous intéresse ?' },
footer: { text: 'Répondez via un bouton' },
header: {
title: 'Menu produits',
hasMediaAttachment: true,
imageMessage: imageMsg,
},
nativeFlowMessage: {
buttons: [
{
name: 'quick_reply',
buttonParamsJson: JSON.stringify({
display_text: '📦 Voir produits',
id: '.produits',
}),
},
{
name: 'ctl_url',
buttonParamsJson: JSON.stringify({
display_text: '🛒 Visiter boutique',
url: 'https://canguidev.fr',
merchant_url: 'https://canguidev.fr',
}),
},
],
},
}),
},
},
};
const msg = generateWAMessageFromContent(jid, content, {});
await sock.relayMessage(jid, msg.message, { messageId: msg.key.id });
}
initBaileys();
app.get('/login', (req, res) => {
res.sendFile(path.join(__dirname, 'public', 'login.html'));
});
app.get('/api/qrcode', (req, res) => {
res.json({ qr: qrData, connected: isConnected });
});
app.post('/sendText', async (req, res) => {
const { phone, message } = req.body;
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
try {
await sock.sendMessage(`${phone}@s.whatsapp.net`, { text: message });
res.json({ success: true });
} catch (e) {
res.status(500).json({ error: e.message });
}
});
app.post('/sendButtons', async (req, res) => {
const { phone } = req.body;
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
try {
const content = {
viewOnceMessage: {
message: {
messageContextInfo: {
deviceListMetadata: {},
deviceListMetadataVersion: 2
},
interactiveMessage: proto.Message.InteractiveMessage.create({
body: { text: "Fizzxy Dev" },
footer: { text: "Bot" },
header: {
title: "Igna",
subtitle: "test",
hasMediaAttachment: false
},
nativeFlowMessage: {
buttons: [
{
name: "single_select",
buttonParamsJson: JSON.stringify({
title: "title",
sections: [
{
title: "menu",
highlight_label: "label",
rows: [
{
header: "header",
title: "title",
description: "description",
id: "id1"
},
{
header: "header",
title: "title",
description: "description",
id: "id2"
}
]
}
]
})
},
{
name: "cta_reply",
buttonParamsJson: JSON.stringify({
display_text: "quick_reply",
id: "message"
})
},
{
name: "cta_url",
buttonParamsJson: JSON.stringify({
display_text: "Visiter Google",
url: "https://www.google.com",
merchant_url: "https://www.google.com"
})
},
{
name: "cta_call",
buttonParamsJson: JSON.stringify({
display_text: "Appeler support",
id: "+33600000000"
})
},
{
name: "cta_copy",
buttonParamsJson: JSON.stringify({
display_text: "Copier code",
id: "codecopie",
copy_code: "123456789"
})
},
{
name: "cta_reminder",
buttonParamsJson: JSON.stringify({
display_text: "Ajouter rappel",
id: "reminder_id"
})
},
{
name: "cta_cancel_reminder",
buttonParamsJson: JSON.stringify({
display_text: "Annuler rappel",
id: "reminder_id"
})
},
{
name: "address_message",
buttonParamsJson: JSON.stringify({
display_text: "Adresse",
id: "adresse_id"
})
},
{
name: "send_location",
buttonParamsJson: ""
}
]
}
})
}
}
};
const msg = generateWAMessageFromContent(`${phone}@s.whatsapp.net`, content, {});
await sock.relayMessage(`${phone}@s.whatsapp.net`, msg.message, { messageId: msg.key.id });
res.json({ success: true });
} catch (e) {
console.error(e);
res.status(500).json({ error: e.message });
}
});
app.listen(3001, () => console.log('🚀 Serveur Baileys démarré sur http://localhost:3001'));