今天初来乍到cnode.js,也应该贡献贡献.看到微信公众平台,开始有点兴奋,能做个机器人玩玩,,随后用Node.js写了一个,觉得其实这没什么意思.很快就觉得腻了,于是有了做发送微信接口的想法.首先要做的我们就要模拟公众平台的登陆.对于微信的这些lib,当然不能直接写在routes里面,,那要怎么办呢?没错,就要封装起来,方便复用.你可以打开控制台看到公众平台的登录请求,还有所需的参数,其中密码它是用它本身的md5进行加密的,那么我们需要做的只是将它copy过来放在一个helpers/wx/md5.js文件里就可以直接用了,以下是微信公众平台解析后格式化的js提交代码
submit: function() {
if (!n()) return;
var e = d.getVal();
t.post("/cgi-bin/login?lang=zh_CN", {
username: e.account,
pwd1: t.md5(e.password.substr(0, 15)),
pwd2: t.md5(e.password),
imgcode: f.data("isHide") ? "": e.verify,
register: e.isRegister,
f: "json"
},
我们要建立一个login的方法
request = require 'superagent'
require __basename + '/helpers/wx/md5'
config = require __basename + '/config/config'
module.exports =
login: (fn) ->
wx_usr = config.wx.user
wx_pwd = md5 config.wx.pwd.substr(0, 16)
request
.post('http://mp.weixin.qq.com/cgi-bin/login?lang=zh_CN')
.type('form')
.send(
username: wx_usr
pwd: wx_pwd
imgcode : ''
f : 'json'
register : 0
)
.end (res) ->
//在这里你已经成功获取cookie了
但是经过分析我想你会发现,这里的cookie其实并非你想要的cookie,因为它包含一些没用的信息Path=,我们设置cookie的时候,事实上是不能用直接设置这样的cookie,应该是一个cookie里面不应该有其他的东西,而分号后面的path应该将它去掉,这里是返回的结果:
[
"mp_user=xxxxxx; Path=/",
"mp_sid=NlJ2Tm5hb1NXRGxOU3V1MzF2a25tSFVWRHhTNkhwek1nMXlEOVZzMnZMUG1lZ29nSkdENGt3WlgwUjBJZnhydndYNkZSd0ZsaHRHdEozSHBIa3QwT3FWTmdXc3RxVFhYUDBCR3dnWkxIRWVvRlZObG15UC83SzU1aEZPZWpocU8=; Path=/"
]
以下是完整的login代码:
login: (fn) ->
wx_usr = config.wx.user
wx_pwd = md5 config.wx.pwd
request
.post('http://mp.weixin.qq.com/cgi-bin/login?lang=zh_CN')
.type('form')
.send(
username: wx_usr
pwd1: wx_pwd
pwd2: wx_pwd
imgcode : ''
f : 'json'
)
.end (res) ->
cookie = ''
for rs in res.header['set-cookie']
cookie += rs.replace(/Path=\//g, '')
fn null, cookie
在这里,我们已经完成登录的操作了,接下来,我们要做的是进行发送,在发送的时候,要把这个cookie设置在请求的地址中,接下来的代码比较简单:(注意:这里面的fakeid是微信公众平台的id,我们可以用控制台去微信公众平台的用户管理中查看,如何获取用户好友的fakeid,接下来一章我会讲)
sender: (options, fn) ->
msg = options.msg
fakeid = options.fakeid
unless msg
fn error: 'missing msg'
return
unless fakeid
fn error: 'missing fakeid'
return
psotParams =
type: 1
content: msg
error: false
tofakeid : fakeid
ajax : 1
request
.post('http://mp.weixin.qq.com/cgi-bin/singlesend?t=ajax-response&lang=zh_CN')
.type('form')
.send(psotParams)
.set('Cookie', options.cookie)
.end (res) ->
fn null, JSON.parse res.text
这里,我们已经能完全发送了,因为返回的结果是一个json,所要最好先JSON.parse一下,里面的成功判断大家可以加上,返回的接口有个叫ret的参数,0为发送成功
{
ret: "0",
msg: "ok"
}
下一章,我会为大家深入讲解怎么发送信息给好友,及获取微信头像等等技术
在调用的时候其实也比较简单,刚忘写了补上:
app.get '/wx/login', (req, res) ->
msg = req.query.msg
wx.login (err, cookie) ->
res.json err if err
data =
msg : msg
fakeid : 'xxxx'
cookie : cookie
wx.sender data, (err, results) ->
res.json err if err
res.json results
sender目前只能实现文字text的发送,但这并不完善,也可以实现图文的发送,但是却必须要上传一张封面图片,上传后需要在发送图文信息的时候把返回结果中的formId
拿到,http://mp.weixin.qq.com/cgi-bin/uploadmaterial?cgi=uploadmaterial&type=2&t=iframe-uploadfile&lang=zh_CN&formId=1
其中的参数type
是上传的类型,语音和视频是0,formId为null.上传封面图片接口如下:
uploadmaterial: (fn) ->
@login (err, cookie) ->
request
.post('http://mp.weixin.qq.com/cgi-bin/uploadmaterial?cgi=uploadmaterial&type=2&t=iframe-uploadfile&lang=zh_CN&formId=1')
.type('form')
.set('Cookie', cookie)
.end (res) ->
results = JSON.parse(res.text).match(/formId, '(\d+)'/)[2]
fn null, results
成功后,可以进行图文消息的发送了.
send_appmsg: (options, fn) ->
@login (err, cookie) ->
psotParams =
error : false
count : 1
AppMsgId : null
title0 : options.title
digest0 : '正文内容'
# content0 : '<p>te<img src="http://www.e-research-solutions.com/system/cms/themes/default/img/top_l.png" /></p><p><span style="color:red">测试标题</span></p>'
content0 : options.msg
fileid0 : '10000001' #此处的id即为封面图片的id
preusername : options.username
ajax : 1
request
.post('http://mp.weixin.qq.com/cgi-bin/operate_appmsg?sub=preview&t=ajax-appmsg-preview')
.type('form')
.set('Cookie', cookie)
.send(psotParams)
.end (res) ->
results = JSON.parse res.text
fn null, results['msg']
@ym1623 支持作者可以换js语法来写,个人很讨厌将js这么好的语言 用coffeescript写,我觉得大多数用node的人都是喜欢js的人 如果写成这样 那还不如去用rails 为嘛用nodejs呢
文件上传怎么体现的?我看这里只有一个post请求,post的数据中并没有封面图片的数据信息?另外,我post了一个文件给微信的这个地址,得到的答案中没有formid,不知道为啥?大侠帮忙解答下?
Line-based text data: text/html <script> document.domain = location.hostname.match(/[^.]?.[^.]?$/); var url = window.location.href, type = url.match(/[?&]type=([^&])/), formId = url.match(/[?&]formId=([^&])/);
type = type[1] || 0;
formId = formId[1];
top.W.upload.err("\344\270\212\344\274\240\347\232\204\347\264\240\346\235\220\346\240\274\345\274\217\344\270\215\346\255\243\347\241\256", type, formId);
</script>
试着将楼主的js用Phpl来实现 不过上传文件一直返回 php代码是这样的
$send_snoopy = new Snoopy();
$send_snoopy->agent = "(Mozilla/5.0 (Windows NT 5.1; rv:19.0) Gecko/20100101 Firefox/19.0)"; //伪装浏览器
$send_snoopy->referer = "http://mp.weixin.qq.com/cgi-bin/indexpage?lang=zh_CN&t=wxm-upload&lang=zh_CN&type=0&fromId=file_from_1341151893625"; //伪装来源页地址 http_referer
$send_snoopy->rawheaders['Cookie'] = $cookie;
$send_snoopy->rawheaders["Host"] = "mp.weixin.qq.com";
$post = array();
$post['uploadfile'] = $imgContent;
$post['formId'] = '';
$submit = "http://mp.weixin.qq.com/cgi-bin/uploadmaterial?cgi=uploadmaterial&type=2&t=iframe-uploadfile&lang=zh_CN&formId=null";
$send_snoopy->submit($submit,$post);
$result = $send_snoopy->results;
preg_match('/formId, \'(\d*)\'/',$result,$imgId);
print_r($result);
echo $imgId[1];
`
<script> document.domain = location.hostname.match(/[^\.]*?\.[^\.]*?$/); var url = window.location.href, type = url.match(/[\?&]type=([^&]*)/), formId = url.match(/[\?&]formId=([^&]*)/);type = type[1] || 0; formId = formId[1]; top.W.upload.err("上传文件失败", type, formId);
</script>`这是为什么呢?
今天开始,好像 http://mp.weixin.qq.com/cgi-bin/singlesend?t=ajax-response&lang=zh_CN 调用不成功了,返回{ret:’-1’,msg:’need post’}
大家要注意一下,因为公众平台本身也有cookie过期时间,大家获取到cookie时应该放入session,另外还需要写个restart.js的脚本,意在解决登录不成功的问题,如果session过期,就运行restart文件,可以写成nodejs也可以写成shell调用.如果大家觉得麻烦的话,大家有时间的话也可以研究一下网页版的微信登录接口,有朋友说用的是line的登录验证方法,已经破解了,大家也可以尝试一下.
@jkeen 我也发现了返回有token,意思是需要连续发起两次登录请求?第一次返回token,第二次加上token再登录?但是经过尝试依旧无法登录,第二次照样返回一个token,而且如果cookie如果没有包含 sig / cert 两个参数,就会返回-6(根据JS的描述是需要验证码),明显太坑爹了!求解决办法
@wengebin 先用http://mp.weixin.qq.com/cgi-bin/verifycode?username=,这个获得验证码,然后用login api去登录,参数里面输入刚才的imgcode,成功后会返回 { "Ret": 302, "ErrMsg": "/cgi-bin/indexpage?t=wxm-index&lang=zh_CN&token=xxxxxx", "ShowVerifyCode": 0, "ErrCode": 0 } 里面的token就是每次发送消息时,需要传给服务器的,每次登录token都会变
@wengebin 我是直接调用 login,从返回的result里取token,好像没有验证码也可以: .send({ username: wx_usr, pwd: wx_pwd, imgcode: '’, f: ‘json’ } 返回: { "Ret": 302, "ErrMsg": "/cgi-bin/indexpage?t=wxm-index&lang=zh_CN&token=xxxxxx", "ShowVerifyCode": 0, "ErrCode": 0 }
var token = res.body.ErrMsg.match(new RegExp("[?&]token=([^&]+)“,"i”));
这个token保存起来,下次访问用保存的cookie和token访问。我的做法是在每次调用接口前调用 /cgi-bin/getcontactinfo? 测试一下cookie和token是否有效。 if (res.statusCode !=200 || res.text.indexOf(‘登录超时’)>0){ //需要重新登录 }else{ 使用之前保存的cookie和token }
最新的代码已经更新至github中:https://github.com/ym1623/node_wx 由于我今天晚上11点钟开始去弄,弄好后马上更新,经测试,现已能正常发送微信 更新如下: 添加:token 添加发送referer防止发送不到信息 本来想弄下记住密码那个功能的,由于时间太晚就没有去更新,大家有谁有兴趣的话可以在git中提交分支给我,我会审阅,希望更完善.
模拟登陆时: { "Ret": 302, "ErrMsg": "/cgi-bin/indexpage?t=wxm-index&lang=zh_CN&token=1741837296", "ShowVerifyCode": 0, "ErrCode": 0 }
发送消息时返回:{"ret":"-20000", "msg":"Request is denied! "},请问什么原因呀?
@huangbz2007 返回cookie:slave_sid=NnY5UjNvN2c5ZldnZWduZ2JJak5qS3N5Qm1zdkhBakw0c0I4UUltX1ZoZjd4QW5HaEh0em9UNF9md3VVNU9FOGZmZTNWQTBRU2RYbU5oMDJZckJrakZqeThMcmN3NWlNaXdOYTVOX1ZfVWhLYmZLNnNudlFZSGxmZ0xxMWp0Nmc=; Path=/; Secure; HttpOnly
cookieVal=slave_sid=NnY5UjNvN2c5ZldnZWduZ2JJak5qS3N5Qm1zdkhBakw0c0I4UUltX1ZoZjd4QW5HaEh0em9UNF9md3VVNU9FOGZmZTNWQTBRU2RYbU5oMDJZckJrakZqeThMcmN3NWlNaXdOYTVOX1ZfVWhLYmZLNnNudlFZSGxmZ0xxMWp0Nmc=; Path=/; Secure; resumeConnection.setRequestProperty("Cookie", cookieVal); resumeConnection .setRequestProperty("Referer", “https://mp.weixin.qq.com/cgi-bin/singlesend?t=ajax-response&lang=zh_CN”);
@huangbz2007 这个是返回的结果,我们模拟登陆后其实是需要这个token和cookie,然后正则过滤一下cookie,得到真正的cookie,你可以在我的github中参考一下看我是怎么做的
我用php写的模拟登录是正常的,发送也正常的,可是我想取得用户的列表就不行了返回 {"ret":"-20000", "msg":"Request is denied! "},昨天的时候还可以,估计是今天微信官方做了调整。大家有什么解决方式吗
你好,我写的代码总返回登录超时,Cookie我 也从模拟登录中得到了,并且请求时也赋值了。还是我给的 formid参数值不对啊 ,你说的那个 formid发送图文时如何获到呢,希望帮忙,谢谢,我的QQ4796642
现在还好使么? headers.set("Referer", “https://mp.weixin.qq.com/cgi-bin/singlemsgpage?fromfakeid=&count=20&t=wxm-singlechat&token=&lang=zh_CN”); headers.set(“User-Agent","Mozilla/5.0 (Windows NT 6.1; rv:22.0) Gecko/20100101 Firefox/22.0”); headers.set(“Content-Type","application/x-www-form-urlencoded; charset=UTF-8”); headers.set(“Host","mp.weixin.qq.com”); headers.set(“Origin","https://mp.weixin.qq.com”);
我都加了这么多了,还是返回 {"ret":"-20000", "msg":"Request is denied! "}