Merge remote-tracking branch 'origin/main'

dev/supplier-email-drawer
Lei OT 9 months ago
commit 724d3df6dc

@ -11,6 +11,7 @@ const {
} = require('@whiskeysockets/baileys'); } = require('@whiskeysockets/baileys');
const { writeFile } = require('fs/promises'); const { writeFile } = require('fs/promises');
const waEmitter = require('../emitter'); const waEmitter = require('../emitter');
const serverConfig = require('../../config').server;
const { formatPhoneNumber, parsePhoneNumber, formatStatus, formatTimestamp } = require('./helper'); const { formatPhoneNumber, parsePhoneNumber, formatStatus, formatTimestamp } = require('./helper');
const generateId = require('../../utils/generateId.util'); const generateId = require('../../utils/generateId.util');
@ -31,7 +32,7 @@ waEmitter.on('connection:close', event => {
}); });
const createWhatsApp = async phone => { const createWhatsApp = async phone => {
let socketRef = null; // let waSocket = null;
let qrCode = null; let qrCode = null;
let connectionStatus = 'offline' let connectionStatus = 'offline'
const channelId = generateId(); const channelId = generateId();
@ -39,60 +40,20 @@ const createWhatsApp = async phone => {
// 储存键值对 msgId-externalId // 储存键值对 msgId-externalId
// TODO 什么时候清理旧的? // TODO 什么时候清理旧的?
const msgIdMap = new Map(); const msgIdMap = new Map();
const logger = P({ timestamp: () => `,"time":"${new Date().toJSON()}"` }, P.destination('./wa-logs-' + phone + '_' + channelId + '.txt'));
const sendTextMessage = async (number, content, externalId) => { logger.level = 'trace';
const jid = formatPhoneNumber(number); const msgRetryCounterCache = new NodeCache();
const storeFilename = './baileys_auth_info/baileys_store_' + phone + '_' + channelId + '.json'
return new Promise(() => { const store = makeInMemoryStore({ logger });
socketRef.sendMessage(jid, { text: content }) store?.readFromFile(storeFilename);
.then(msg => { // save every 10s
msgIdMap.set(msg.key.id, externalId); setInterval(() => {
}) store?.writeToFile(storeFilename);
.catch(ex => { }, 10_000);
waEmitter.emit('message:updated', { const { state, saveCreds } = await useMultiFileAuthState('baileys_auth_info/' + phone + '_' + channelId);
id: generateId(), // fetch latest version of WA Web
externalId, const { version, isLatest } = await fetchLatestBaileysVersion();
status: 'failed', const waVersion = version.join('.') + ', ' + (isLatest ? 'latest' : 'out');
direction: 'outbound',
from: whatsAppNo,
to: number,
errro: `发送文本消息出错 ` + ex,
eventSource: 'WA-01-HK.sendMessage.promise.catch',
updateTime: formatTimestamp(new Date().getTime() / 1000),
});
});
});
};
const sendImageMessage = async (number, imageUrl) => {
const jid = formatPhoneNumber(number);
try {
const msgInfo = await socketRef.sendMessage(jid, {
image: { url: imageUrl },
});
return {
messageId: msgInfo?.key?.id ?? generateId()
};
} catch (ex) {
waEmitter.emit('message.error', {
messge: `[${whatsAppNo}->${number}]发送图片消息出错`,
from: whatsAppNo,
to: number,
error: ex
})
console.error(`[${whatsAppNo}->${number}]发送图片消息出错: `, ex);
}
};
const getProfilePicture = async (whatsAppNo) => {
const number = formatPhoneNumber(whatsAppNo);
try {
const ppUrl = await socketRef.profilePictureUrl(number);
console.log('头像: ' + ppUrl);
} catch (ex) {
console.error('头像出错: ', ex);
}
}
const handleMessagesUpsert = async upsert => { const handleMessagesUpsert = async upsert => {
console.info('messages.upsert: ', JSON.stringify(upsert, undefined, 2)); console.info('messages.upsert: ', JSON.stringify(upsert, undefined, 2));
@ -139,7 +100,7 @@ const createWhatsApp = async phone => {
}, },
whatsAppNo, whatsAppNo,
fromMe: msg.key.fromMe, fromMe: msg.key.fromMe,
eventSource: 'WA-01-HK.messages.upsert.notify', eventSource: serverConfig.name + '.messages.upsert.notify',
updateTime: formatTimestamp(msg.messageTimestamp), updateTime: formatTimestamp(msg.messageTimestamp),
}); });
} else { } else {
@ -163,7 +124,7 @@ const createWhatsApp = async phone => {
}, },
whatsAppNo, whatsAppNo,
fromMe: msg.key.fromMe, fromMe: msg.key.fromMe,
eventSource: 'WA-01-HK.messages.upsert.notify', eventSource: serverConfig.name + '.messages.upsert.notify',
createTime: formatTimestamp(msg.messageTimestamp), createTime: formatTimestamp(msg.messageTimestamp),
}); });
} }
@ -197,7 +158,7 @@ const createWhatsApp = async phone => {
}, },
whatsAppNo, whatsAppNo,
fromMe: msg.key.fromMe, fromMe: msg.key.fromMe,
eventSource: 'WA-01-HK.messages.upsert.append', eventSource: serverConfig.name + '.messages.upsert.append',
updateTime: formatTimestamp(msg.messageTimestamp), updateTime: formatTimestamp(msg.messageTimestamp),
}); });
} }
@ -228,32 +189,16 @@ const createWhatsApp = async phone => {
}, },
whatsAppNo, whatsAppNo,
fromMe: msg.key.fromMe, fromMe: msg.key.fromMe,
eventSource: 'WA-01-HK.messages.updated', eventSource: serverConfig.name + '.messages.updated',
updateTime: formatTimestamp(new Date().getTime() / 1000), updateTime: formatTimestamp(new Date().getTime() / 1000),
}); });
} }
} }
const handleConnection = async (socket) => { const handleCredsUpdate = async () => {
socketRef = socket; await saveCreds();
socketRef.sendMessage('8613317835586@s.whatsapp.net', { text: '原始连接上了:' + new Date().toString()})
} }
const start = () => {
const start = async () => {
const logger = P({ timestamp: () => `,"time":"${new Date().toJSON()}"` }, P.destination('./wa-logs-' + phone + '_' + channelId + '.txt'));
logger.level = 'trace';
const msgRetryCounterCache = new NodeCache();
const storeFilename = './baileys_auth_info/baileys_store_' + phone + '_' + channelId + '.json'
const store = makeInMemoryStore({ logger });
store?.readFromFile(storeFilename);
// save every 10s
setInterval(() => {
store?.writeToFile(storeFilename);
}, 10_000);
const { state, saveCreds } = await useMultiFileAuthState('baileys_auth_info/' + phone + '_' + channelId);
// fetch latest version of WA Web
const { version, isLatest } = await fetchLatestBaileysVersion();
const waSocket = makeWASocket({ const waSocket = makeWASocket({
version, version,
@ -268,7 +213,7 @@ const createWhatsApp = async phone => {
//retryRequestDelayMs: 1000*25, //retryRequestDelayMs: 1000*25,
// https://github.com/WhiskeySockets/Baileys/blob/31bc8ab/src/Utils/generics.ts#L21 // https://github.com/WhiskeySockets/Baileys/blob/31bc8ab/src/Utils/generics.ts#L21
// https://github.com/WhiskeySockets/Baileys/blob/31bc8ab4e2c825c0d774875701ed07e20d05bdb6/WAProto/WAProto.proto // https://github.com/WhiskeySockets/Baileys/blob/31bc8ab4e2c825c0d774875701ed07e20d05bdb6/WAProto/WAProto.proto
browser: Browsers.ubuntu('IOS_PHONE'),//Browsers.macOS('SAFARI'),//Browsers.ubuntu('IOS_PHONE'),//Browsers.baileys('WEAR_OS'),// browser: Browsers.macOS('SAFARI'),//Browsers.macOS('SAFARI'),//Browsers.ubuntu('IOS_PHONE'),//Browsers.baileys('WEAR_OS'),//
msgRetryCounterCache, msgRetryCounterCache,
generateHighQualityLinkPreview: false, generateHighQualityLinkPreview: false,
syncFullHistory: false, syncFullHistory: false,
@ -276,79 +221,136 @@ const createWhatsApp = async phone => {
store?.bind(waSocket.ev); store?.bind(waSocket.ev);
waSocket.ev.process( const sendTextMessage = async (number, content, externalId) => {
// events is a map for event name => event data const jid = formatPhoneNumber(number);
async(events) => { waSocket.sendMessage(jid, { text: content })
// something about the connection changed .then(msg => {
// maybe it closed, or we received all offline message or connection opened msgIdMap.set(msg.key.id, externalId);
if(events['connection.update']) { })
const update = events['connection.update'] .catch(ex => {
const { connection, qr, lastDisconnect } = update console.error('sendTextMessage.error: ', ex)
waEmitter.emit('message:updated', {
id: generateId(),
externalId,
status: 'failed',
direction: 'outbound',
from: whatsAppNo,
to: number,
error: `发送文本消息出错 ` + ex,
eventSource: serverConfig.name + '.sendMessage.promise.catch',
updateTime: formatTimestamp(new Date().getTime() / 1000),
});
});
};
const sendImageMessage = async (number, imageUrl) => {
const jid = formatPhoneNumber(number);
try {
const msgInfo = await waSocket.sendMessage(jid, {
image: { url: imageUrl },
});
return {
messageId: msgInfo?.key?.id ?? generateId()
};
} catch (ex) {
waEmitter.emit('message.error', {
messge: `[${whatsAppNo}->${number}]发送图片消息出错`,
from: whatsAppNo,
to: number,
error: ex
})
console.error(`[${whatsAppNo}->${number}]发送图片消息出错: `, ex);
}
};
if (connection === 'close') { const getProfilePicture = async (whatsAppNo) => {
if((lastDisconnect?.error)?.output?.statusCode !== DisconnectReason.loggedOut) { const number = formatPhoneNumber(whatsAppNo);
start(); try {
} else { const ppUrl = await waSocket.profilePictureUrl(number);
waEmitter.emit('connection:close', { console.log('头像: ' + ppUrl);
whatsAppNo, channelId, } catch (ex) {
eventSource: 'WA-01-HK.connection.update.close', console.error('头像出错: ', ex);
status: 'offline', }
}); }
} waSocket.ev.on('connection.update', async update => {
} else if (connection === 'open') { console.log('connection update: ', update);
handleConnection(waSocket) const { connection, lastDisconnect, qr } = update;
connectionStatus = 'open';
waEmitter.emit('connection:open', {
status: 'open', whatsAppNo, channelId,
eventSource: 'WA-01-HK.connection.update.open',
socket: waSocket,
sendTextMessage,
sendImageMessage,
});
} else if (qr !== undefined) {
// WebSocket 创建成功等待扫码,如果没有扫码会更新 qr
// 第一次一分钟,后面是 20 秒更新一次
if (qrCode === null) {
qrCode = qr;
console.info('qr: ', qr)
waEmitter.emit('connection:qr', { if (connection === 'close') {
createTimestamp: Date.now(), if((lastDisconnect?.error)?.output?.statusCode !== DisconnectReason.loggedOut) {
status: 'offline', start();
version: '0.111111', } else {
channelId: channelId,
phone: phone, waEmitter.emit('connection:close', {
qrCode: qr, whatsAppNo, channelId,
socket: waSocket eventSource: serverConfig.name + '.connection.update.close',
}) status: 'offline',
} else { });
// 第一次二维码时效后退出,不需要等待更新二维码 }
waSocket.logout(() => '二维码已过期'); } else if (connection === 'open') {
} connectionStatus = 'open';
} waEmitter.emit('connection:open', {
} status: 'open', whatsAppNo, channelId,
eventSource: serverConfig.name + '.connection.update.open',
});
waEmitter.on('request.' + whatsAppNo + '.send.message', event => {
// const jid = formatPhoneNumber(event.to);
const {to: number, externalId, content} = event;
console.info('request.' + whatsAppNo + '.send.message:', event)
waSocket.sendMessage(
number + '@s.whatsapp.net',
{ text: content}
).then(msg => {
msgIdMap.set(msg.key.id, externalId);
}).catch(ex => {
console.error('sendMessage.error: ', ex)
waEmitter.emit('message:updated', {
id: generateId(),
externalId,
status: 'failed',
direction: 'outbound',
from: whatsAppNo,
to: number,
error: `发送文本消息出错 ` + ex,
eventSource: serverConfig.name + '.sendMessage.catch',
updateTime: formatTimestamp(new Date().getTime() / 1000),
});
});
//sendTextMessage(number, content, externalId);
});
} else if (qr !== undefined) {
// WebSocket 创建成功等待扫码,如果没有扫码会更新 qr
// 第一次一分钟,后面是 20 秒更新一次
if (qrCode === null) {
qrCode = qr;
console.info('qr: ', qr);
waEmitter.emit('creds:update', {
id: generateId(),
qr, whatsAppNo,
server:serverConfig.name,
eventSource: 'creds.update',
createTime: formatTimestamp(new Date().getTime() / 1000),
});
} else {
// 第一次二维码时效后退出,不需要等待更新二维码
waSocket.logout(() => '二维码已过期');
}
if (events['creds.update']) { }
await saveCreds() });
}
}
)
waSocket.ev.on('creds.update', async () => await saveCreds()); waSocket.ev.on('creds.update', handleCredsUpdate);
waSocket.ev.on('messages.upsert', handleMessagesUpsert); waSocket.ev.on('messages.upsert', handleMessagesUpsert);
waSocket.ev.on('messages.update', handleMessagesUpdate); waSocket.ev.on('messages.update', handleMessagesUpdate);
}; };
return { return {
createTimestamp: Date.now(), createTimestamp: Date.now(),
status: 'offline', status: 'offline',
version: 'no-promise-version', version: waVersion,
channelId: channelId, channelId: channelId,
phone: phone, phone: phone,
start, start,
sendTextMessage,
sendImageMessage,
getProfilePicture,
}; };
}; };

Loading…
Cancel
Save