dsdsdds
This commit is contained in:
parent
d77d5724c3
commit
30a6f2171d
54
Dockerfile
54
Dockerfile
@ -1,46 +1,28 @@
|
|||||||
FROM node:20-slim
|
FROM node:20
|
||||||
|
|
||||||
# Install system dependencies for Venom-bot, Puppeteer and Sharp
|
# Installation des dépendances système nécessaires à sharp (et puppeteer)
|
||||||
RUN apt-get update && apt-get install -y \
|
RUN apt-get update && apt-get install -y \
|
||||||
libvips-dev \
|
libvips-dev \
|
||||||
libnss3 \
|
libnss3 \
|
||||||
libatk-bridge2.0-0 \
|
libatk-bridge2.0-0 \
|
||||||
libxss1 \
|
libxss1 \
|
||||||
libasound2 \
|
libasound2 \
|
||||||
libgtk-3-0 \
|
libgtk-3-0 \
|
||||||
libgbm1 \
|
libgbm1 \
|
||||||
libx11-xcb1 \
|
libx11-xcb1 \
|
||||||
libxcomposite1 \
|
libxcomposite1 \
|
||||||
libxdamage1 \
|
libxdamage1 \
|
||||||
libxrandr2 \
|
libxrandr2 \
|
||||||
xdg-utils \
|
xdg-utils \
|
||||||
fonts-liberation \
|
--no-install-recommends && \
|
||||||
libappindicator3-1 \
|
apt-get clean && rm -rf /var/lib/apt/lists/*
|
||||||
libnspr4 \
|
|
||||||
lsb-release \
|
|
||||||
wget \
|
|
||||||
# Clean up
|
|
||||||
&& apt-get clean \
|
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
|
||||||
|
|
||||||
# Set working directory
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
# Install Chromium manually (recommended approach for Venom-bot)
|
|
||||||
ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=true
|
|
||||||
ENV PUPPETEER_EXECUTABLE_PATH=/usr/bin/chromium
|
|
||||||
|
|
||||||
# Copy package files first for better layer caching
|
|
||||||
COPY package.json yarn.lock ./
|
COPY package.json yarn.lock ./
|
||||||
|
RUN yarn install
|
||||||
|
|
||||||
# Install project dependencies
|
|
||||||
RUN yarn install --frozen-lockfile --production=false
|
|
||||||
|
|
||||||
# Copy application files
|
|
||||||
COPY . .
|
COPY . .
|
||||||
|
|
||||||
# Expose the application port
|
|
||||||
EXPOSE 3001
|
EXPOSE 3001
|
||||||
|
CMD ["yarn", "start"]
|
||||||
# Start command
|
|
||||||
CMD ["yarn", "start"]
|
|
||||||
|
|||||||
355
index.js
355
index.js
@ -1,5 +1,6 @@
|
|||||||
const venom = require('venom-bot');
|
const { default: makeWASocket, useMultiFileAuthState, DisconnectReason, fetchLatestBaileysVersion, proto, generateWAMessageFromContent, generateWAMessageContent ,prepareWAMessageMedia} = require('@fizzxydev/baileys-pro');
|
||||||
const express = require('express');
|
const express = require('express');
|
||||||
|
const { Boom } = require('@hapi/boom');
|
||||||
const qrcode = require('qrcode');
|
const qrcode = require('qrcode');
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
const fs = require('fs');
|
const fs = require('fs');
|
||||||
@ -8,42 +9,44 @@ const app = express();
|
|||||||
app.use(express.json());
|
app.use(express.json());
|
||||||
app.use(express.static('public'));
|
app.use(express.static('public'));
|
||||||
|
|
||||||
let client;
|
let sock;
|
||||||
let qrData = null;
|
let qrData = null;
|
||||||
let isConnected = false;
|
let isConnected = false;
|
||||||
|
|
||||||
// Initialisation Venom-bot
|
const initBaileys = async () => {
|
||||||
const initVenom = async () => {
|
const { version } = await fetchLatestBaileysVersion();
|
||||||
client = await venom.create({
|
const { state, saveCreds } = await useMultiFileAuthState('auth');
|
||||||
session: 'whatsapp-session',
|
|
||||||
disableSpins: true,
|
sock = makeWASocket({
|
||||||
logQR: false,
|
version,
|
||||||
catchQR: (base64Qrimg) => {
|
auth: state,
|
||||||
qrcode.toDataURL(base64Qrimg, (err, url) => {
|
printQRInTerminal: false
|
||||||
if (err) {
|
});
|
||||||
console.error('Erreur QR code:', err);
|
|
||||||
return;
|
sock.ev.on('connection.update', async (update) => {
|
||||||
}
|
const { connection, lastDisconnect, qr } = update;
|
||||||
qrData = url;
|
if (qr) {
|
||||||
isConnected = false;
|
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;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
client.onStateChange((state) => {
|
sock.ev.on('creds.update', saveCreds);
|
||||||
console.log('État de connexion:', state);
|
|
||||||
isConnected = state === 'CONNECTED';
|
|
||||||
|
|
||||||
if (state === 'DISCONNECTED') {
|
|
||||||
console.log('🔁 Tentative de reconnexion...');
|
|
||||||
setTimeout(initVenom, 5000);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
initVenom();
|
initBaileys();
|
||||||
|
|
||||||
// Routes Express
|
|
||||||
app.use('/static', express.static(path.join(__dirname, 'public')));
|
app.use('/static', express.static(path.join(__dirname, 'public')));
|
||||||
|
|
||||||
app.get('/login', (req, res) => {
|
app.get('/login', (req, res) => {
|
||||||
@ -56,10 +59,9 @@ app.get('/api/qrcode', (req, res) => {
|
|||||||
|
|
||||||
app.post('/sendText', async (req, res) => {
|
app.post('/sendText', async (req, res) => {
|
||||||
const { phone, message } = req.body;
|
const { phone, message } = req.body;
|
||||||
if (!client || !isConnected) return res.status(400).json({ error: 'Non connecté' });
|
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await client.sendText(`${phone}@c.us`, message);
|
await sock.sendMessage(`${phone}@s.whatsapp.net`, { text: message });
|
||||||
res.json({ success: true });
|
res.json({ success: true });
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
res.status(500).json({ error: e.message });
|
res.status(500).json({ error: e.message });
|
||||||
@ -68,100 +70,255 @@ app.post('/sendText', async (req, res) => {
|
|||||||
|
|
||||||
app.post('/sendButtons', async (req, res) => {
|
app.post('/sendButtons', async (req, res) => {
|
||||||
const { phone } = req.body;
|
const { phone } = req.body;
|
||||||
if (!client || !isConnected) return res.status(400).json({ error: 'Non connecté' });
|
if (!sock || !isConnected) return res.status(400).json({ error: 'Non connecté' });
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const buttons = [
|
const content = {
|
||||||
{ body: "📩 Contacter support" },
|
message: {
|
||||||
{ body: "🌐 Voir notre site" },
|
interactiveMessage: proto.Message.InteractiveMessage.create({
|
||||||
{ body: "📞 Appeler le support" }
|
body: { text: "Bienvenue sur notre service !" },
|
||||||
];
|
footer: { text: "Choisis une action ci-dessous" },
|
||||||
|
header: {
|
||||||
await client.sendButtons(
|
title: "Menu principal",
|
||||||
`${phone}@c.us`,
|
hasMediaAttachment: false
|
||||||
"Bienvenue sur notre service !",
|
},
|
||||||
buttons,
|
nativeFlowMessage: {
|
||||||
"Choisis une action ci-dessous"
|
buttons: [
|
||||||
);
|
{
|
||||||
|
name: "cta_reply",
|
||||||
res.json({ success: true });
|
buttonParamsJson: JSON.stringify({
|
||||||
} catch (e) {
|
display_text: "📩 Contacter support",
|
||||||
console.error('❌ Erreur boutons:', e);
|
id: "support_action"
|
||||||
res.status(500).json({ error: e.message });
|
})
|
||||||
}
|
},
|
||||||
});
|
{
|
||||||
|
name: "cta_url",
|
||||||
app.post('/sendInteractiveImage', async (req, res) => {
|
buttonParamsJson: JSON.stringify({
|
||||||
const { phone, caption, footer } = req.body;
|
display_text: "🌐 Voir notre site",
|
||||||
if (!client || !isConnected) return res.status(400).json({ error: 'Non connecté' });
|
url: "https://canguidev.fr",
|
||||||
|
merchant_url: "https://canguidev.fr"
|
||||||
try {
|
})
|
||||||
const imagePath = path.join(__dirname, 'public', 'logo-merlo-cs-FR.jpg');
|
},
|
||||||
const imageBuffer = fs.readFileSync(imagePath);
|
{
|
||||||
|
name: "cta_call",
|
||||||
await client.sendImage(
|
buttonParamsJson: JSON.stringify({
|
||||||
`${phone}@c.us`,
|
display_text: "📞 Appeler le support",
|
||||||
imageBuffer,
|
id: "+33612345678"
|
||||||
'logo.jpg',
|
})
|
||||||
caption || 'Description par défaut',
|
}
|
||||||
{
|
]
|
||||||
footer: footer || 'Pied de page',
|
}
|
||||||
buttons: [
|
})
|
||||||
{ buttonId: 'id1', buttonText: { displayText: '📄 Voir proposition' }, type: 1 },
|
|
||||||
{ buttonId: 'id2', buttonText: { displayText: '🔧 Spécifications' }, type: 1 }
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
);
|
};
|
||||||
|
|
||||||
|
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 });
|
res.json({ success: true });
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error('❌ Erreur image interactive:', e);
|
console.error('❌ Erreur bouton actif :', e);
|
||||||
res.status(500).json({ error: e.message });
|
res.status(500).json({ error: e.message });
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
// Votre route POST
|
||||||
|
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é à WhatsApp' });
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const BASE_URL = process.env.BASE_URL || 'https://wa.canguidev.fr';
|
||||||
|
const imageUrl = `${BASE_URL}/static/logo-merlo-cs-FR.jpg`;
|
||||||
|
|
||||||
|
// Construire le Header correct
|
||||||
|
const header = proto.Message.InteractiveMessage.create({
|
||||||
|
// ici on injecte l'image dans le header sans besoin de type explicite
|
||||||
|
header: proto.Message.InteractiveMessage.Header.create({
|
||||||
|
type: proto.Message.InteractiveMessage.HeaderType.IMAGE,
|
||||||
|
imageMessage: { url: imageUrl },
|
||||||
|
title: 'Igna',
|
||||||
|
subtitle: 'test'
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
// Body et Footer
|
||||||
|
const body = proto.Message.InteractiveMessage.Body.create({
|
||||||
|
text: caption || 'Description par défaut'
|
||||||
|
});
|
||||||
|
const foot = proto.Message.InteractiveMessage.Footer.create({
|
||||||
|
text: footer || 'Pied de page'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Vos boutons URL
|
||||||
|
const nativeFlow = proto.Message.InteractiveMessage.NativeFlowMessage.create({
|
||||||
|
buttons: [
|
||||||
|
{
|
||||||
|
name: 'cta_url',
|
||||||
|
buttonParamsJson: JSON.stringify({
|
||||||
|
display_text: '📄 Voir proposition',
|
||||||
|
url: 'https://merlo-ch.com/uploads/proposition/f_p_250505_0000136_00008_EB00001909.pdf'
|
||||||
|
})
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'cta_url',
|
||||||
|
buttonParamsJson: JSON.stringify({
|
||||||
|
display_text: '🔧 Spécifications',
|
||||||
|
url: 'https://merlo-ch.com/uploads/proposition/d_p_250505_0000136_00008_EB00001909.pdf'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
]
|
||||||
|
});
|
||||||
|
|
||||||
|
// Construire l'InteractiveMessage complet
|
||||||
|
const interactiveMsg = proto.Message.InteractiveMessage.create({
|
||||||
|
header,
|
||||||
|
body,
|
||||||
|
footer: foot,
|
||||||
|
nativeFlowMessage: nativeFlow
|
||||||
|
});
|
||||||
|
|
||||||
|
// Envelopper (ici dans viewOnceMessage, comme dans votre exemple)
|
||||||
|
const raw = {
|
||||||
|
viewOnceMessage: {
|
||||||
|
message: {
|
||||||
|
messageContextInfo: {
|
||||||
|
deviceListMetadata: {},
|
||||||
|
deviceListMetadataVersion: 2
|
||||||
|
},
|
||||||
|
interactiveMessage: interactiveMsg
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Générer et relayer
|
||||||
|
const jid = `${phone}@s.whatsapp.net`;
|
||||||
|
const msg = generateWAMessageFromContent(jid, raw, {});
|
||||||
|
await sock.relayMessage(jid, msg.message, { messageId: msg.key.id });
|
||||||
|
|
||||||
|
return res.json({ success: true });
|
||||||
|
} catch (e) {
|
||||||
|
console.error('❌ Erreur /sendInteractiveImage :', e);
|
||||||
|
return res.status(500).json({ error: e.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 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 });
|
||||||
|
// }
|
||||||
|
// });
|
||||||
app.post('/sendProductMessage', async (req, res) => {
|
app.post('/sendProductMessage', async (req, res) => {
|
||||||
if (!client || !isConnected) return res.status(400).json({ error: 'Non connecté' });
|
// On vérifie sock et isConnected, pas client
|
||||||
|
if (!sock || !isConnected) {
|
||||||
|
return res.status(400).json({ error: 'Non connecté à WhatsApp' });
|
||||||
|
}
|
||||||
|
|
||||||
const {
|
const {
|
||||||
phone,
|
phone,
|
||||||
productImageUrl,
|
productImageUrl,
|
||||||
|
productImageCount,
|
||||||
productTitle,
|
productTitle,
|
||||||
productDescription,
|
productDescription,
|
||||||
price,
|
priceAmount1000,
|
||||||
productUrl
|
currencyCode,
|
||||||
|
retailerId,
|
||||||
|
productUrl,
|
||||||
|
businessOwnerJid,
|
||||||
|
caption,
|
||||||
|
messageTitle,
|
||||||
|
footer,
|
||||||
|
interactiveButtons = [],
|
||||||
|
quoted
|
||||||
} = req.body;
|
} = req.body;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Venom-bot n'a pas de méthode native pour les produits
|
const jid = `${phone}@s.whatsapp.net`;
|
||||||
// On simule avec une image et des boutons
|
|
||||||
const buttons = [
|
|
||||||
{ buttonId: 'buy', buttonText: { displayText: '🛒 Acheter' }, type: 1 },
|
|
||||||
{ buttonId: 'details', buttonText: { displayText: 'ℹ️ Détails' }, type: 1 }
|
|
||||||
];
|
|
||||||
|
|
||||||
const caption = `*${productTitle}*\n\n${productDescription}\n\nPrix: ${price}`;
|
// Build product payload exactly comme avant
|
||||||
|
const productPayload = {
|
||||||
|
productImage: { url: productImageUrl },
|
||||||
|
productImageCount,
|
||||||
|
title: productTitle, // ici productTitle
|
||||||
|
description: productDescription,
|
||||||
|
priceAmount1000: priceAmount1000 * 1000, // ajustez si besoin
|
||||||
|
currencyCode,
|
||||||
|
retailerId,
|
||||||
|
url: productUrl
|
||||||
|
};
|
||||||
|
|
||||||
await client.sendImage(
|
// Transformer les boutons
|
||||||
`${phone}@c.us`,
|
const buttons = interactiveButtons.map(btn => {
|
||||||
productImageUrl,
|
const params = {};
|
||||||
'product.jpg',
|
if (btn.id) params.id = btn.id;
|
||||||
caption,
|
if (btn.url) params.url = btn.url;
|
||||||
{ buttons }
|
if (btn.display_text) params.display_text = btn.display_text;
|
||||||
|
return {
|
||||||
|
name: btn.name,
|
||||||
|
buttonParamsJson: JSON.stringify(params)
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
// **On utilise sock.sendMessage** et non client.sendMessage
|
||||||
|
await sock.sendMessage(
|
||||||
|
jid,
|
||||||
|
{
|
||||||
|
product: productPayload,
|
||||||
|
businessOwnerJid,
|
||||||
|
caption,
|
||||||
|
title: messageTitle,
|
||||||
|
footer,
|
||||||
|
media: true,
|
||||||
|
interactiveButtons: buttons
|
||||||
|
},
|
||||||
|
quoted ? { quoted } : {}
|
||||||
);
|
);
|
||||||
|
|
||||||
res.json({ success: true });
|
return res.json({ success: true });
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error('❌ Erreur produit:', e);
|
console.error('❌ Erreur /sendProductMessage :', e);
|
||||||
res.status(500).json({ error: e.message });
|
return res.status(500).json({ error: e.message });
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Gestion des erreurs
|
|
||||||
|
// 5) 404 et gestion des erreurs
|
||||||
app.use((req, res) => res.status(404).json({ error: 'Ressource introuvable' }));
|
app.use((req, res) => res.status(404).json({ error: 'Ressource introuvable' }));
|
||||||
app.use((err, req, res, next) => {
|
app.use((err, req, res, next) => {
|
||||||
console.error('Erreur:', err);
|
console.error('Middleware d’erreur :', err);
|
||||||
res.status(500).json({ error: err.message });
|
res.status(500).json({ error: err.message });
|
||||||
});
|
});
|
||||||
|
app.listen(3001, () => console.log('🚀 Serveur Baileys démarré sur http://localhost:3001'));
|
||||||
app.listen(3001, () => console.log('🚀 Serveur Venom démarré sur http://localhost:3001'));
|
|
||||||
|
|||||||
44
package.json
44
package.json
@ -1,37 +1,17 @@
|
|||||||
{
|
{
|
||||||
"name": "whatsapp-api-venom",
|
"name": "baileys-whatsapp-api",
|
||||||
"version": "2.0.1",
|
"version": "1.0.0",
|
||||||
"description": "API WhatsApp avec Venom-bot - Solution complète de messagerie",
|
"description": "API WhatsApp avec Baileys Pro",
|
||||||
"main": "server.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "node server.js",
|
"start": "node index.js"
|
||||||
"dev": "nodemon server.js",
|
|
||||||
"test": "jest --coverage",
|
|
||||||
"lint": "eslint .",
|
|
||||||
"format": "prettier --write .",
|
|
||||||
"postinstall": "node node_modules/venom-bot/dist/install/install-chromium.js"
|
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"venom-bot": "^4.3.7",
|
"@fizzxydev/baileys-pro": "latest",
|
||||||
"express": "^4.18.2",
|
"@hapi/boom": "^10.0.1",
|
||||||
"express-rate-limit": "^6.8.1",
|
"axios": "^1.4.0",
|
||||||
"helmet": "^7.1.0",
|
"express": "^4.18.4",
|
||||||
"morgan": "^1.10.0",
|
"qrcode": "^1.5.1",
|
||||||
"qrcode": "^1.5.3",
|
"sharp": "^0.33.0"
|
||||||
"winston": "^3.11.0",
|
|
||||||
"axios": "^1.6.2",
|
|
||||||
"puppeteer": "^21.9.0"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"eslint": "^8.56.0",
|
|
||||||
"eslint-config-prettier": "^9.1.0",
|
|
||||||
"jest": "^29.7.0",
|
|
||||||
"nodemon": "^3.0.2",
|
|
||||||
"prettier": "^3.1.1",
|
|
||||||
"supertest": "^6.3.3"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">=16.0.0",
|
|
||||||
"npm": ">=8.0.0"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user