This repository has been archived by the owner on Nov 20, 2019. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 89
/
Copy pathdemo.js
407 lines (373 loc) · 13.5 KB
/
demo.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
'use strict'
const log4js = require('log4js')
const Padchat = require('./index')
const fs = require('fs')
const util = require('util')
const qrcode = require('qrcode-terminal')
/**
* 创建日志目录
*/
try {
require('fs').mkdirSync('./logs')
} catch (e) {
if (e.code !== 'EEXIST') {
console.error('Could not set up log directory, error: ', e)
process.exit(1)
}
}
try {
log4js.configure('./log4js.json')
} catch (e) {
console.error('载入log4js日志输出配置错误: ', e)
process.exit(1);
}
const logger = log4js.getLogger('app')
const dLog = log4js.getLogger('dev')
logger.info('demo start!')
const autoData = {
wxData: '',
token : '',
}
let server = ''
server = 'ws://127.0.0.1:7777'
server = 'ws://padchat.server/{YourToken}'
// 服务器地址及token请联系客服获取
try {
const tmpBuf = fs.readFileSync('./config.json')
const data = JSON.parse(String(tmpBuf))
autoData.wxData = data.wxData
autoData.token = data.token
logger.info('载入设备参数与自动登陆数据:%o ', autoData)
} catch (e) {
logger.warn('没有在本地发现设备登录参数或解析数据失败!如首次登录请忽略!')
}
const wx = new Padchat(server)
logger.info('当前连接接口服务器为:', server)
let disconnectCount = 0 // 断开计数
let connected = false // 成功连接标志
wx
.on('close', (code, msg) => {
// 需要注意关闭代码为3201-3203的错误,重连也用,具体参考文档中`close`事件说明
if (code > 3200) {
logger.warn(`Websocket已关闭!code: ${code} - ${msg}`)
return
}
logger.info(`Websocket已关闭!code: ${code} - ${msg}`)
// 根据是否成功连接过判断本次是未能连接成功还是与服务器连接中断
if (connected) {
connected = false
disconnectCount++
logger.info(`第 ${disconnectCount} 次与服务器连接断开!现在将重试连接服务器。`)
} else {
logger.debug(`未能连接服务器!将重试连接服务器。`)
}
// 重新启动websocket连接
wx.start()
})
.on('open', async () => {
let ret
logger.info('连接成功!')
connected = true
// 非首次登录时最好使用以前成功登录时使用的设备参数,
// 否则可能会被tx服务器怀疑账号被盗,导致手机端被登出
ret = await wx.init()
if (!ret.success) {
logger.error('新建任务失败!', ret)
return
}
logger.info('新建任务成功, json: ', ret)
//先尝试使用断线重连方式登陆
if (autoData.token) {
ret = await wx.login('auto', autoData)
if (ret.success) {
logger.info('断线重连请求成功!', ret)
return
}
logger.warn('断线重连请求失败!', ret)
ret = await wx.login('request', autoData)
if (ret.success) {
logger.info('自动登录请求成功!', ret)
return
}
logger.warn('自动登录请求失败!', ret)
}
ret = await wx.login('qrcode')
if (!ret.success) {
logger.error('使用qrcode登录模式失败!', ret)
return
}
logger.info('使用qrcode登录模式!')
})
.on('qrcode', data => {
// 如果存在url,则直接在终端中生成二维码并显示
if (data.url) {
logger.info(`登陆二维码内容为: "${data.url}",请使用微信扫描下方二维码登陆!`)
qrcode.generate(data.url, { small: true })
} else {
logger.error(`未能获得登陆二维码!`)
}
})
.on('scan', data => {
switch (data.status) {
case 0:
logger.info('等待扫码...', data)
break;
case 1:
// {
// status : 1,
// expiredTime: 239,
// headUrl : 'http://wx.qlogo.cn/mmhead/ver_1/xxxxxxx/0', //头像url
// nickName : '木匠' //昵称
// }
logger.info('已扫码,请在手机端确认登陆...', data)
break;
case 2:
// {
// password : '***hide***', // 可忽略
// status : 2,
// expiredTime: 238,
// headUrl : 'http://wx.qlogo.cn/mmhead/ver_1/xxxxxxx/0', //头像url
// subStatus : 0 // 登陆操作状态码
// 以下字段仅在登录成功时有效
// external : '1',
// email : '',
// uin : 149806460, // 微信账号uin,全局唯一
// deviceType : 'android', // 登陆的主设备类型
// nickName : '木匠' //昵称
// userName : 'wxid_xxxxxx', // 微信账号id,全局唯一
// phoneNumber: '18012345678', // 微信账号绑定的手机号
// }
switch (data.subStatus) {
case 0:
logger.info('扫码成功!登陆成功!', data)
break;
case 1:
logger.info('扫码成功!登陆失败!', data)
break;
default:
logger.info('扫码成功!未知状态码!', data)
break;
}
break;
// 如果等待登陆超时或手机上点击了取消登陆,需要重新调用登陆
case 3:
logger.info('二维码已过期!请重新调用登陆接口!', data)
break;
case 4:
logger.info('手机端已取消登陆!请重新调用登陆接口!', data)
break;
default:
logger.warn('未知登陆状态!请重新调用登陆接口!', data)
break;
}
})
.on('login', async () => {
logger.info('微信账号登陆成功!')
let ret
ret = await wx.getMyInfo()
logger.info('当前账号信息:', ret.data)
// 如果不想同步通讯录信息,可关闭同步通讯录
// await wx.setSyncContact(false)
// 主动同步通讯录
await wx.syncContact()
await saveAutoData()
})
.on('autoLogin', async () => {
// 自动重连后需要保存新的自动登陆数据
await saveAutoData()
})
.on('logout', ({ msg }) => {
logger.info('微信账号已退出!', msg)
})
.on('over', ({ msg }) => {
logger.info('任务实例已关闭!', msg)
})
.on('loaded', async () => {
logger.info('通讯录同步完毕!')
// 主动触发同步消息
await wx.syncMsg()
const ret = await wx.sendMsg('filehelper', '你登录了!')
logger.info('发送信息结果:', ret)
})
.on('sns', (data, msg) => {
logger.info('收到朋友圈事件!请查看朋友圈新消息哦!', msg)
})
.on('contact', async data => {
logger.info('收到推送联系人:%s - %s\n', data.userName, data.nickName, JSON.stringify(data))
})
.on('push', async data => {
// 消息类型 data.mType
// 1 文字消息
// 3 收到图片消息
// 34 语音消息
// 35 用户头像buf
// 37 收到好友请求消息
// 42 名片消息
// 43 视频消息
// 47 表情消息
// 48 定位消息
// 49 APP消息(文件 或者 链接 H5)
// 50 语音通话
// 51 状态通知(如打开与好友/群的聊天界面)
// 52 语音通话通知
// 53 语音通话邀请
// 62 小视频
// 2000 转账消息
// 2001 收到红包消息
// 3000 群邀请
// 9999 系统通知
// 10000 微信通知信息. 微信群信息变更通知,多为群名修改,进群,离群信息,不包含群内聊天信息
// 10002 撤回消息
// --------------------------------
// 注意,如果是来自微信群的消息,data.content字段中包含发言人的wxid及其发言内容,需要自行提取
// 各类复杂消息,data.content中是xml格式的文本内容,需要自行从中提取各类数据。(如好友请求)
let rawFile
switch (data.mType) {
case 3:
logger.info('收到来自 %s 的图片消息,包含图片数据:%s,xml内容:\n%s', data.fromUser, !!data.data, data.content)
rawFile = data.data || null
logger.info('图片缩略图数据base64尺寸:%d', rawFile.length)
await wx.getMsgImage(data)
.then(ret => {
rawFile = ret.data.image || ''
logger.info('获取消息原始图片结果:%s, 获得图片base64尺寸:%d', ret.success, rawFile.length)
})
logger.info('图片数据base64尺寸:%d', rawFile.length)
await wx.sendImage('filehelper', rawFile)
.then(ret => {
logger.info('转发图片信息给 %s 结果:', 'filehelper', ret)
})
.catch(e => {
logger.warn('转发图片信息异常:', e.message)
})
break
case 43:
logger.info('收到来自 %s 的视频消息,包含视频数据:%s,xml内容:\n%s', data.fromUser, !!data.data, data.content)
rawFile = data.data || null
if (!rawFile) {
await wx.getMsgVideo(data)
.then(ret => {
rawFile = ret.data.video || ''
logger.info('获取消息原始视频结果:%s, 获得视频base64尺寸:%d', ret.success, rawFile.length)
})
}
logger.info('视频数据base64尺寸:%d', rawFile.length)
break
case 1:
if (data.fromUser === 'newsapp') { // 腾讯新闻发的信息太长
break
}
logger.info('收到来自 %s 的文本消息:', data.fromUser, data.description || data.content)
if (/ding/.test(data.content)) {
await wx.sendMsg(data.fromUser, 'dong. receive:' + data.content)
.then(ret => {
logger.info('回复信息给%s 结果:', data.fromUser, ret)
})
.catch(e => {
logger.warn('回复信息异常:', e.message)
})
} else if (/^#.*/.test(data.content) || /^[\w]*:\n#.*/.test(data.content)) {
await onMsg(data)
.catch(e => {
logger.warn('处理信息异常:', e)
})
}
break
case 34:
logger.info('收到来自 %s 的语音消息,包含语音数据:%s,xml内容:\n%s', data.fromUser, !!data.data, data.content)
// 超过30Kb的语音数据不会包含在推送信息中,需要主动拉取
rawFile = data.data || null
if (!rawFile) {
// BUG: 超过60Kb的语音数据,只能拉取到60Kb,也就是说大约36~40秒以上的语音会丢失后边部分语音内容
await wx.getMsgVoice(data)
.then(ret => {
rawFile = ret.data.voice || ''
logger.info('获取消息原始语音结果:%s, 获得语音base64尺寸:%d,拉取到数据尺寸:%d', ret.success, rawFile.length, ret.data.size)
})
}
logger.info('语音数据base64尺寸:%d', rawFile.length)
if (rawFile.length > 0) {
let match = data.content.match(/length="(\d+)"/) || []
const length = match[1] || 0
match = data.content.match(/voicelength="(\d+)"/) || []
const ms = match[1] || 0
logger.info('语音数据语音长度:%d ms,xml内记录尺寸:%d', ms, length)
await wx.sendVoice('filehelper', rawFile, ms)
.then(ret => {
logger.info('转发语音信息给 %s 结果:', 'filehelper', ret)
})
.catch(e => {
logger.warn('转发语音信息异常:', e.message)
})
}
break
case 49:
logger.info('收到一条来自 %s 的appmsg富媒体消息:', data.fromUser, data)
break
case 10002:
if (data.fromUser === 'weixin') {
//每次登陆,会收到一条系统垃圾推送,过滤掉
break
}
logger.info('用户 %s 撤回了一条消息:', data.fromUser, data)
break
default:
logger.info('收到推送消息:', data)
break
}
})
.on('error', e => {
logger.error('ws 错误:', e.message)
})
.on('warn', e => {
logger.error('任务出现错误:', e.message)
})
/**
* @description 保存自动登陆数据
*/
async function saveAutoData() {
let ret = await wx.getWxData()
if (!ret.success) {
logger.warn('获取设备参数未成功! json:', ret)
return
}
logger.info('获取设备参数成功, json: ', ret)
Object.assign(autoData, { wxData: ret.data.wxData })
ret = await wx.getLoginToken()
if (!ret.success) {
logger.warn('获取自动登陆数据未成功! json:', ret)
return
}
logger.info('获取自动登陆数据成功, json: ', ret)
Object.assign(autoData, { token: ret.data.token })
// NOTE: 这里将设备参数保存到本地,以后再次登录此账号时提供相同参数
fs.writeFileSync('./config.json', JSON.stringify(autoData, null, 2))
logger.info('设备参数已写入到 ./config.json文件')
}
async function onMsg(data) {
const content = data.content.replace(/^[\w:\n]*#/m, '')
let [cmd, ...args] = content.split('\n')
args = args.map(str => {
try {
str = JSON.parse(str)
} catch (e) {
}
return str
})
if (cmd && wx[cmd] && typeof wx[cmd] === 'function') {
logger.info('执行函数 %s,参数:', cmd, args)
await wx[cmd](...args)
.then(ret => {
logger.info('执行函数 %s 结果:%o', cmd, ret)
})
.catch(e => {
logger.warn('执行函数 %s 异常:', e)
})
}
}
process.on('uncaughtException', e => {
logger.error('Main', 'uncaughtException:', e)
})
process.on('unhandledRejection', e => {
logger.error('Main', 'unhandledRejection:', e)
})