whatsapp/index.js

157 lines
4.5 KiB
JavaScript
Raw Normal View History

2025-05-06 14:04:11 +00:00
const express = require('express');
const qrcode = require('qrcode');
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:16:28 +00:00
generateWAMessageFromContent,
proto
2025-05-06 20:12:39 +00:00
} = require('@whiskeysockets/baileys');
2025-05-06 20:21:45 +00:00
const { Boom } = require('@hapi/boom');
2025-05-06 20:12:39 +00:00
2025-05-06 14:04:11 +00:00
const app = express();
app.use(express.json());
2025-05-06 20:24:10 +00:00
const path = require('path'); // à ajouter en haut si pas encore importé
// Sert les fichiers statiques depuis le dossier "public"
app.use(express.static('public'));
// Route pour afficher la page de login avec QR code
app.get('/login', (req, res) => {
res.sendFile(path.join(__dirname, 'public', 'login.html'));
});
2025-05-06 08:51:13 +00:00
2025-05-06 18:21:47 +00:00
let sock;
2025-05-06 20:21:45 +00:00
let qrCodeData = null;
2025-05-06 14:04:11 +00:00
let isConnected = false;
2025-05-06 08:51:13 +00:00
2025-05-06 20:21:45 +00:00
async function startSock() {
const { state, saveCreds } = await useMultiFileAuthState('auth_info');
2025-05-06 18:21:47 +00:00
const { version } = await fetchLatestBaileysVersion();
sock = makeWASocket({
version,
auth: state,
2025-05-06 20:21:45 +00:00
printQRInTerminal: true
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 18:21:47 +00:00
if (qr) {
2025-05-06 20:21:45 +00:00
qrCodeData = await qrcode.toDataURL(qr);
2025-05-06 18:21:47 +00:00
}
2025-05-06 19:25:43 +00:00
2025-05-06 18:21:47 +00:00
if (connection === 'close') {
2025-05-06 20:21:45 +00:00
const shouldReconnect = (lastDisconnect?.error instanceof Boom)
&& lastDisconnect.error.output.statusCode !== DisconnectReason.loggedOut;
console.log('❌ Déconnecté. Reconnexion :', shouldReconnect);
if (shouldReconnect) startSock();
2025-05-06 18:21:47 +00:00
} else if (connection === 'open') {
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 10:12:55 +00:00
2025-05-06 18:21:47 +00:00
sock.ev.on('creds.update', saveCreds);
2025-05-06 20:21:45 +00:00
}
2025-05-06 19:25:43 +00:00
2025-05-06 20:21:45 +00:00
startSock();
2025-05-06 09:54:23 +00:00
app.get('/api/qrcode', (req, res) => {
2025-05-06 20:21:45 +00:00
res.json({ qr: qrCodeData, connected: isConnected });
2025-05-06 14:04:11 +00:00
});
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;
2025-05-06 18:21:47 +00:00
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
2025-05-06 20:21:45 +00:00
2025-05-06 09:38:23 +00:00
try {
2025-05-06 18:21:47 +00:00
await sock.sendMessage(`${phone}@s.whatsapp.net`, { text: message });
2025-05-06 14:04:11 +00:00
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 20:21:45 +00:00
app.post('/sendImage', async (req, res) => {
const { phone, imageUrl, caption } = req.body;
2025-05-06 20:16:28 +00:00
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
2025-05-06 19:31:03 +00:00
2025-05-06 20:16:28 +00:00
try {
2025-05-06 20:21:45 +00:00
await sock.sendMessage(`${phone}@s.whatsapp.net`, {
image: { url: imageUrl },
caption: caption || ''
});
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
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 19:14:46 +00:00
2025-05-06 20:21:45 +00:00
app.post('/sendDocument', async (req, res) => {
const { phone, documentUrl, fileName, mimetype } = req.body;
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
try {
await sock.sendMessage(`${phone}@s.whatsapp.net`, {
document: { url: documentUrl },
fileName: fileName || 'fichier.pdf',
mimetype: mimetype || 'application/pdf'
});
res.json({ success: true });
} catch (e) {
res.status(500).json({ error: e.message });
}
});
app.post('/sendInteractive', async (req, res) => {
2025-05-06 19:51:17 +00:00
const { phone } = req.body;
2025-05-06 20:21:45 +00:00
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
try {
const jid = `${phone}@s.whatsapp.net`;
const content = {
viewOnceMessage: {
message: {
messageContextInfo: {
deviceListMetadata: {},
deviceListMetadataVersion: 2
2025-05-06 19:54:47 +00:00
},
2025-05-06 20:21:45 +00:00
interactiveMessage: proto.Message.InteractiveMessage.create({
body: { text: 'Bienvenue ! Choisissez une action :' },
footer: { text: 'Service WhatsApp' },
nativeFlowMessage: {
buttons: [
{
name: 'cta_reply',
buttonParamsJson: JSON.stringify({
display_text: '✅ Confirmer',
id: 'confirm_action'
})
},
{
name: 'cta_url',
buttonParamsJson: JSON.stringify({
display_text: '🌐 Site Web',
url: 'https://canguidev.fr'
})
}
]
}
})
}
2025-05-06 19:54:47 +00:00
}
2025-05-06 20:21:45 +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 20:21:45 +00:00
2025-05-06 20:16:28 +00:00
res.json({ success: true });
} catch (e) {
2025-05-06 20:21:45 +00:00
console.error('❌ Erreur interactive :', e);
2025-05-06 20:16:28 +00:00
res.status(500).json({ error: e.message });
}
2025-05-06 19:54:47 +00:00
});
2025-05-06 20:21:45 +00:00
const PORT = process.env.PORT || 3001;
app.listen(PORT, () => {
console.log(`🚀 Serveur prêt sur http://localhost:${PORT}`);
2025-05-06 15:34:25 +00:00
});