This commit is contained in:
cangui 2025-05-06 22:21:45 +02:00
parent 3aca1bbbe2
commit dcb36bf059

209
index.js
View File

@ -1,54 +1,42 @@
const express = require('express'); const express = require('express');
const qrcode = require('qrcode'); const qrcode = require('qrcode');
const path = require('path');
const fs = require('fs');
const NodeCache = require('node-cache');
const { const {
makeWASocket, default: makeWASocket,
useMultiFileAuthState, useMultiFileAuthState,
DisconnectReason, DisconnectReason,
fetchLatestBaileysVersion, fetchLatestBaileysVersion,
generateWAMessageFromContent, generateWAMessageFromContent,
proto proto
} = require('@whiskeysockets/baileys'); } = require('@whiskeysockets/baileys');
const { Boom } = require('@hapi/boom');
const app = express(); const app = express();
app.use(express.json()); app.use(express.json());
app.use(express.static('public'));
let sock; let sock;
let qrData = null; let qrCodeData = null;
let isConnected = false; let isConnected = false;
const groupCache = new NodeCache({ stdTTL: 300, useClones: false }); async function startSock() {
const { state, saveCreds } = await useMultiFileAuthState('auth_info');
async function getMessageFromStore(key) {
return { conversation: "Message temporaire pour retry" };
}
const initBaileys = async () => {
const { version } = await fetchLatestBaileysVersion(); const { version } = await fetchLatestBaileysVersion();
const { state, saveCreds } = await useMultiFileAuthState('auth');
sock = makeWASocket({ sock = makeWASocket({
version, version,
auth: state, auth: state,
markOnlineOnConnect: false, printQRInTerminal: true
getMessage: async (key) => await getMessageFromStore(key),
cachedGroupMetadata: async (jid) => groupCache.get(jid)
}); });
sock.ev.on('connection.update', async ({ connection, lastDisconnect, qr }) => { sock.ev.on('connection.update', async ({ connection, lastDisconnect, qr }) => {
if (qr) { if (qr) {
qrData = await qrcode.toDataURL(qr); qrCodeData = await qrcode.toDataURL(qr);
isConnected = false;
} }
if (connection === 'close') { if (connection === 'close') {
const shouldReconnect = lastDisconnect?.error?.output?.statusCode !== DisconnectReason.loggedOut; const shouldReconnect = (lastDisconnect?.error instanceof Boom)
console.log(shouldReconnect ? '🔁 Reconnexion...' : '❌ Déconnecté.'); && lastDisconnect.error.output.statusCode !== DisconnectReason.loggedOut;
if (shouldReconnect) initBaileys(); console.log('❌ Déconnecté. Reconnexion :', shouldReconnect);
if (shouldReconnect) startSock();
} else if (connection === 'open') { } else if (connection === 'open') {
console.log('✅ Connecté à WhatsApp'); console.log('✅ Connecté à WhatsApp');
isConnected = true; isConnected = true;
@ -56,28 +44,17 @@ const initBaileys = async () => {
}); });
sock.ev.on('creds.update', saveCreds); sock.ev.on('creds.update', saveCreds);
}
sock.ev.on('messages.upsert', async ({ messages }) => { startSock();
const msg = messages[0];
if (!msg.key.fromMe && msg.message?.conversation) {
console.log('💬 Message reçu de', msg.key.remoteJid, ':', msg.message.conversation);
}
});
};
initBaileys();
app.get('/login', (req, res) => {
res.sendFile(path.join(__dirname, 'public', 'login.html'));
});
app.get('/api/qrcode', (req, res) => { app.get('/api/qrcode', (req, res) => {
res.json({ qr: qrData, connected: isConnected }); res.json({ qr: qrCodeData, connected: isConnected });
}); });
app.post('/sendText', async (req, res) => { app.post('/sendText', async (req, res) => {
const { phone, message } = req.body; const { phone, message } = req.body;
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' }); if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
try { try {
await sock.sendMessage(`${phone}@s.whatsapp.net`, { text: message }); await sock.sendMessage(`${phone}@s.whatsapp.net`, { text: message });
res.json({ success: true }); res.json({ success: true });
@ -86,101 +63,85 @@ app.post('/sendText', async (req, res) => {
} }
}); });
app.post('/sendButtons', async (req, res) => { app.post('/sendImage', async (req, res) => {
const { phone, imageUrl, caption } = req.body;
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
try {
await sock.sendMessage(`${phone}@s.whatsapp.net`, {
image: { url: imageUrl },
caption: caption || ''
});
res.json({ success: true });
} catch (e) {
res.status(500).json({ error: e.message });
}
});
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) => {
const { phone } = req.body; const { phone } = req.body;
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' }); if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
const jid = `${phone}@s.whatsapp.net`;
const imageUrl = 'https://wa.canguidev.fr/static/logo-merlo-cs-FR.jpg';
const content = {
viewOnceMessage: {
message: {
messageContextInfo: {
deviceListMetadata: {},
deviceListMetadataVersion: 2
},
interactiveMessage: proto.Message.InteractiveMessage.create({
header: {
hasMediaAttachment: true,
imageMessage: { url: imageUrl }
},
body: { text: 'Bienvenue !' },
footer: { text: 'Choisissez une action :' },
nativeFlowMessage: {
buttons: [
{
name: 'cta_reply',
buttonParamsJson: JSON.stringify({ display_text: 'Répondre', id: 'reply_id' })
},
{
name: 'cta_url',
buttonParamsJson: JSON.stringify({ display_text: 'Site Web', url: 'https://canguidev.fr' })
}
]
}
})
}
}
};
try { try {
const jid = `${phone}@s.whatsapp.net`;
const content = {
viewOnceMessage: {
message: {
messageContextInfo: {
deviceListMetadata: {},
deviceListMetadataVersion: 2
},
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'
})
}
]
}
})
}
}
};
const msg = generateWAMessageFromContent(jid, content, {}); const msg = generateWAMessageFromContent(jid, content, {});
await sock.relayMessage(jid, msg.message, { messageId: msg.key.id }); await sock.relayMessage(jid, msg.message, { messageId: msg.key.id });
res.json({ success: true }); res.json({ success: true });
} catch (e) { } catch (e) {
console.error('❌ Erreur sendButtons :', e); console.error('❌ Erreur interactive :', e);
res.status(500).json({ error: e.message }); res.status(500).json({ error: e.message });
} }
}); });
const PORT = process.env.PORT || 3001;
app.post('/sendInteractiveImage', async (req, res) => { app.listen(PORT, () => {
const { phone } = req.body; console.log(`🚀 Serveur prêt sur http://localhost:${PORT}`);
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté à WhatsApp' });
const jid = `${phone}@s.whatsapp.net`;
const imageUrl = 'https://wa.canguidev.fr/static/logo-merlo-cs-FR.jpg';
const content = {
viewOnceMessage: {
message: {
messageContextInfo: {
deviceListMetadata: {},
deviceListMetadataVersion: 2
},
interactiveMessage: proto.Message.InteractiveMessage.create({
header: {
hasMediaAttachment: true,
imageMessage: { url: imageUrl }
},
body: { text: '📄 Document interactif' },
footer: { text: 'Sélectionne une option :' },
nativeFlowMessage: {
buttons: [
{
name: 'cta_url',
buttonParamsJson: JSON.stringify({ display_text: 'Voir PDF', url: 'https://example.com/doc.pdf' })
}
]
}
})
}
}
};
try {
const msg = generateWAMessageFromContent(jid, content, {});
await sock.relayMessage(jid, msg.message, { messageId: msg.key.id });
res.json({ success: true });
} catch (e) {
console.error('❌ Erreur interactive image :', e);
res.status(500).json({ error: e.message });
}
}); });
app.use((req, res) => res.status(404).json({ error: 'Ressource introuvable' }));
app.use((err, req, res, next) => {
console.error('❌ Middleware erreur :', err);
res.status(500).json({ error: err.message });
});
app.listen(3001, () => console.log('🚀 Serveur Baileys opérationnel sur http://localhost:3001'));