2025-05-06 13:54:09 +00:00
|
|
|
|
// index.js
|
|
|
|
|
|
const {
|
|
|
|
|
|
default: makeWASocket,
|
|
|
|
|
|
useMultiFileAuthState,
|
|
|
|
|
|
DisconnectReason,
|
|
|
|
|
|
fetchLatestBaileysVersion,
|
|
|
|
|
|
proto,
|
|
|
|
|
|
generateWAMessageFromContent
|
|
|
|
|
|
} = 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 axios = require('axios')
|
2025-05-06 10:12:55 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +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 13:54:09 +00:00
|
|
|
|
let sock
|
|
|
|
|
|
let qrData = null
|
|
|
|
|
|
let isConnected = false
|
|
|
|
|
|
|
|
|
|
|
|
// Dictionnaire des URLs PDF
|
|
|
|
|
|
const PDF_LINKS = {
|
|
|
|
|
|
prop: 'https://merlo-ch.com/uploads/proposition/f_p_250505_0000136_00008_EB00001909.pdf',
|
|
|
|
|
|
spec: 'https://merlo-ch.com/uploads/proposition/d_p_250505_0000136_00008_EB00001909.pdf'
|
|
|
|
|
|
}
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 10:12:55 +00:00
|
|
|
|
const initBaileys = async () => {
|
2025-05-06 13:54:09 +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 13:54:09 +00:00
|
|
|
|
})
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
sock.ev.on('connection.update', update => {
|
|
|
|
|
|
const { connection, lastDisconnect, qr } = update
|
2025-05-06 10:12:55 +00:00
|
|
|
|
if (qr) {
|
2025-05-06 13:54:09 +00:00
|
|
|
|
qrData = qrcode.toDataURL(qr)
|
|
|
|
|
|
isConnected = false
|
2025-05-06 10:12:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
if (connection === 'close') {
|
2025-05-06 13:54:09 +00:00
|
|
|
|
const shouldReconnect = lastDisconnect?.error?.output?.statusCode !== DisconnectReason.loggedOut
|
|
|
|
|
|
console.log(shouldReconnect ? '🔁 Reconnexion...' : '❌ Déconnecté.')
|
|
|
|
|
|
if (shouldReconnect) initBaileys()
|
2025-05-06 10:12:55 +00:00
|
|
|
|
} else if (connection === 'open') {
|
2025-05-06 13:54:09 +00:00
|
|
|
|
console.log('✅ Connecté à WhatsApp')
|
|
|
|
|
|
isConnected = true
|
2025-05-06 10:12:55 +00:00
|
|
|
|
}
|
2025-05-06 13:54:09 +00:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
sock.ev.on('creds.update', saveCreds)
|
|
|
|
|
|
|
|
|
|
|
|
// Gestion du clic sur les boutons pour envoyer le PDF
|
|
|
|
|
|
sock.ev.on('messages.upsert', async ({ messages }) => {
|
|
|
|
|
|
const msg = messages[0]
|
|
|
|
|
|
if (!msg.key.fromMe && msg.message?.buttonsResponseMessage) {
|
|
|
|
|
|
const btnId = msg.message.buttonsResponseMessage.selectedButtonId
|
|
|
|
|
|
const pdfUrl = PDF_LINKS[btnId]
|
|
|
|
|
|
if (!pdfUrl) return
|
2025-05-06 10:12:55 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
try {
|
|
|
|
|
|
const resp = await axios.get(pdfUrl, { responseType: 'arraybuffer' })
|
|
|
|
|
|
const pdfBuffer = Buffer.from(resp.data)
|
|
|
|
|
|
|
|
|
|
|
|
await sock.sendMessage(msg.key.remoteJid, {
|
|
|
|
|
|
document: pdfBuffer,
|
|
|
|
|
|
fileName: btnId === 'prop' ? 'Proposition.pdf' : 'Spec_machine.pdf',
|
|
|
|
|
|
mimetype: 'application/pdf'
|
|
|
|
|
|
})
|
|
|
|
|
|
} catch (e) {
|
|
|
|
|
|
console.error('❌ Erreur envoi PDF :', e)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
initBaileys().catch(console.error)
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
// Affichage du QR code pour login
|
2025-05-06 09:54:23 +00:00
|
|
|
|
app.get('/login', (req, res) => {
|
2025-05-06 13:54:09 +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 13:54:09 +00:00
|
|
|
|
res.json({ qr: qrData, connected: isConnected })
|
|
|
|
|
|
})
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
// Envoi d'un simple texte
|
2025-05-06 09:38:23 +00:00
|
|
|
|
app.post('/sendText', async (req, res) => {
|
2025-05-06 13:54:09 +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 13:54:09 +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 13:54:09 +00:00
|
|
|
|
res.status(500).json({ error: e.message })
|
2025-05-06 09:38:23 +00:00
|
|
|
|
}
|
2025-05-06 13:54:09 +00:00
|
|
|
|
})
|
2025-05-06 09:43:48 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
// Envoi de boutons sans image
|
2025-05-06 09:54:23 +00:00
|
|
|
|
app.post('/sendButtons', async (req, res) => {
|
2025-05-06 13:54:09 +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 13:54:09 +00:00
|
|
|
|
header: proto.Message.InteractiveMessage.Header.create({
|
2025-05-06 12:28:20 +00:00
|
|
|
|
title: "Menu principal",
|
|
|
|
|
|
hasMediaAttachment: false
|
2025-05-06 13:54:09 +00:00
|
|
|
|
}),
|
|
|
|
|
|
body: proto.Message.InteractiveMessage.Body.create({
|
|
|
|
|
|
text: "Bienvenue sur notre service !"
|
|
|
|
|
|
}),
|
|
|
|
|
|
footer: proto.Message.InteractiveMessage.Footer.create({
|
|
|
|
|
|
text: "Choisis une action ci-dessous"
|
|
|
|
|
|
}),
|
|
|
|
|
|
nativeFlowMessage: proto.Message.InteractiveMessage.NativeFlowMessage.create({
|
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 13:54:09 +00:00
|
|
|
|
})
|
2025-05-06 12:28:20 +00:00
|
|
|
|
})
|
2025-05-06 12:00:21 +00:00
|
|
|
|
}
|
2025-05-06 13:54:09 +00:00
|
|
|
|
}
|
2025-05-06 12:08:18 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
const msg = generateWAMessageFromContent(
|
|
|
|
|
|
`${phone}@s.whatsapp.net`,
|
|
|
|
|
|
content,
|
|
|
|
|
|
{}
|
|
|
|
|
|
)
|
|
|
|
|
|
await sock.relayMessage(
|
|
|
|
|
|
msg.key.remoteJid,
|
|
|
|
|
|
msg.message,
|
|
|
|
|
|
{ messageId: msg.key.id }
|
|
|
|
|
|
)
|
2025-05-06 11:53:28 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
res.json({ success: true })
|
2025-05-06 11:53:28 +00:00
|
|
|
|
} catch (e) {
|
2025-05-06 13:54:09 +00:00
|
|
|
|
console.error('❌ Erreur sendButtons :', e)
|
|
|
|
|
|
res.status(500).json({ error: e.message })
|
2025-05-06 11:53:28 +00:00
|
|
|
|
}
|
2025-05-06 13:54:09 +00:00
|
|
|
|
})
|
2025-05-06 08:51:13 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
// Envoi de l'image interactive avec CTA URL
|
2025-05-06 12:17:18 +00:00
|
|
|
|
app.post('/sendInteractiveImage', async (req, res) => {
|
2025-05-06 13:54:09 +00:00
|
|
|
|
const {
|
|
|
|
|
|
phone,
|
|
|
|
|
|
caption = 'Description par défaut',
|
|
|
|
|
|
title = 'Titre par défaut',
|
|
|
|
|
|
footer = 'Pied de page',
|
|
|
|
|
|
propositionUrl,
|
|
|
|
|
|
specUrl
|
|
|
|
|
|
} = 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 13:54:09 +00:00
|
|
|
|
const jid = `${phone}@s.whatsapp.net`
|
|
|
|
|
|
const imagePath = path.join(__dirname, 'public', 'logo-merlo-cs-FR.jpg')
|
|
|
|
|
|
const imgBuffer = fs.readFileSync(imagePath)
|
|
|
|
|
|
|
|
|
|
|
|
// Construction du native interactive message
|
|
|
|
|
|
const interactiveContent = proto.Message.InteractiveMessage.create({
|
|
|
|
|
|
header: proto.Message.InteractiveMessage.Header.create({
|
|
|
|
|
|
title,
|
|
|
|
|
|
hasMediaAttachment: true
|
|
|
|
|
|
}),
|
|
|
|
|
|
body: proto.Message.InteractiveMessage.Body.create({
|
|
|
|
|
|
text: caption
|
|
|
|
|
|
}),
|
|
|
|
|
|
footer: proto.Message.InteractiveMessage.Footer.create({
|
|
|
|
|
|
text: footer
|
|
|
|
|
|
}),
|
|
|
|
|
|
nativeFlowMessage: proto.Message.InteractiveMessage.NativeFlowMessage.create({
|
|
|
|
|
|
buttons: [
|
|
|
|
|
|
{
|
|
|
|
|
|
name: 'prop',
|
|
|
|
|
|
buttonParamsJson: JSON.stringify({
|
|
|
|
|
|
display_text: 'Proposition',
|
|
|
|
|
|
url: propositionUrl,
|
|
|
|
|
|
merchant_url: propositionUrl
|
|
|
|
|
|
})
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
name: 'spec',
|
|
|
|
|
|
buttonParamsJson: JSON.stringify({
|
|
|
|
|
|
display_text: 'Spec machine',
|
|
|
|
|
|
url: specUrl,
|
|
|
|
|
|
merchant_url: specUrl
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
})
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// On emballe l’image en tant que miniature (viewOnce)
|
|
|
|
|
|
const messageContent = {
|
|
|
|
|
|
viewOnceMessage: {
|
|
|
|
|
|
message: {
|
|
|
|
|
|
messageContextInfo: {
|
|
|
|
|
|
deviceListMetadata: {},
|
|
|
|
|
|
deviceListMetadataVersion: 2
|
|
|
|
|
|
},
|
|
|
|
|
|
viewOnceMessage: {
|
|
|
|
|
|
message: {
|
|
|
|
|
|
imageMessage: {
|
|
|
|
|
|
mimetype: 'image/jpeg',
|
|
|
|
|
|
jpegThumbnail: imgBuffer
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
interactiveMessage: interactiveContent
|
2025-05-06 12:17:18 +00:00
|
|
|
|
}
|
2025-05-06 13:54:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const waMsg = generateWAMessageFromContent(jid, messageContent, {})
|
|
|
|
|
|
await sock.relayMessage(
|
|
|
|
|
|
waMsg.key.remoteJid,
|
|
|
|
|
|
waMsg.message,
|
|
|
|
|
|
{ messageId: waMsg.key.id }
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
res.json({ success: true })
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
console.error('❌ Erreur sendInteractiveImage:', err)
|
|
|
|
|
|
res.status(500).json({ error: err.message })
|
2025-05-06 12:17:18 +00:00
|
|
|
|
}
|
2025-05-06 13:54:09 +00:00
|
|
|
|
})
|
2025-05-06 12:17:18 +00:00
|
|
|
|
|
2025-05-06 13:54:09 +00:00
|
|
|
|
const PORT = process.env.PORT || 3001
|
|
|
|
|
|
app.listen(PORT, () => {
|
|
|
|
|
|
console.log(`🚀 Serveur Baileys démarré sur http://localhost:${PORT}`)
|
|
|
|
|
|
})
|