--- layout: post title: 快速自制微信图片机器人 tags: [微信, 图片, Pixiv, 机器人, PHP] --- 优化真的是很复杂啊…… # 起因 前段时间,我做出来了[能发图片的机器人](/2021/02/19/picbot.html),做出来之后我拿给群友们体验,但是很遗憾的是那个代码实在是不太行,首先有2个问题,第一是微信获取`access_token`的次数是有限的,我的第一版代码在每一次调用都去获取`access_token`,这样很快次数就会消耗光,后来我稍微改进了一下,设置了个缓存,结果呢,我检测的时候用了次数更少的接口……简直是太蠢了……之后呢?结果今天发现代码里有两个获取`access_token`的地方,缓存完全没起到作用…… 总之上面的问题各种波折总算是解决好了,然后还有一个问题是我的图片来源是[Lolicon API](https://api.lolicon.app/setu),然后调用限制是300次/天,说实话,对于一个人来说这个数量是够了,但是如果有很多人,像测试号最多能容纳100人,那每天每人也就只有3次调用的机会。 那要怎么解决调用次数的问题呢?我首先想的就是缓存结果。 # 解决API调用次数过少的问题 因为对于图片来说,基本上没有什么变化的信息,所以如果能将每一次的结果缓存的话其实也没有问题。所以说干就干,我单独开了一个仓库[pixiv-index](https://github.com/Mabbs/pixiv-index)用来存储缓存的结果,具体代码的话都在这个仓库里面,每天会调用那个API直到用完次数。 考虑到大多数情况下也不需要原图,所以这个API里的图片都只是长或宽最大为1200px的缩略图。 使用方法也很简单,像PHP的话就可以这样写: ```php array('method' => 'GET','header' => "referer: https://www.pixiv.net/")))); $retry-=1; } $context = stream_context_create(array( 'http' => array( 'method' => 'POST', 'header' => 'Content-Type: multipart/form-data; boundary='.MULTIPART_BOUNDARY, 'content' => "--".MULTIPART_BOUNDARY."\r\n". "Content-Disposition: filename=\"image.jpg\"\r\n". "Content-Type: image/jpg\r\n\r\n". $picdata."\r\n". "--".MULTIPART_BOUNDARY."--\r\n" ) )); file_get_contents('https://api.weixin.qq.com/cgi-bin/message/custom/send?access_token='.$_SESSION['access_token'] , false, stream_context_create(array('http' => array('method' => 'POST','header' => 'Content-type: application/json;charset=utf-8','content' => '{ "touser":"'.$_GET["openid"].'", "msgtype":"image", "image": { "media_id":"'.json_decode(file_get_contents('https://api.weixin.qq.com/cgi-bin/media/upload?access_token='.$_SESSION['access_token'].'&type=image', false, $context),true)['media_id'].'" } }')))); exit(); } $timestamp=$_GET["timestamp"]; $nonce=$_GET["nonce"]; $tmpArr=array($token, $timestamp, $nonce); sort($tmpArr, SORT_STRING); if( sha1(implode($tmpArr)) == $_GET["signature"] ){ if($_GET["echostr"]){ echo $_GET["echostr"]; }else{ // 加载XML内容 $content = file_get_contents("php://input"); $p = xml_parser_create(); xml_parse_into_struct($p, $content, $vals, $index); xml_parser_free($p); if($vals[$index['MSGTYPE'][0]]['value'] == 'text'){ if($vals[$index['CONTENT'][0]]['value'] == '来点色图'){ echo ' '.time().' '; file_get_contents('https://'.$_SERVER['HTTP_HOST'].$_SERVER['PHP_SELF'].'?upap=1&openid='.$vals[$index['FROMUSERNAME'][0]]['value'], false, stream_context_create(array('http' => array('timeout' => 0.5)))); }else{ echo 'success'; } } } }else{ echo 'error'; } ``` 2021.02.26更新:似乎在库中的图片有一些是被删掉了,所以为了提高回复的成功率,增加了3次重试。 # 如何使用? 具体应该不需要我说了吧,看之前的几篇关于微信机器人的文章里面的这段就行了。这里我删掉了2个参数,又增加了2个,一个是Token,想填啥都行,只要和测试号里配置一样就行。另一个是source,那个是Pixiv的图片服务器,如果后端服务器在国外那这个就不用管了,如果在国内的话需要改成`https://i.pixiv.cat`来做反代,或者如果有其他反代服务也可以,自己用CloudFlare Worker建一个也没有问题。 # 结尾 那个Lolicon API实在是不好用,不过我也懒得解决了,所以就托学弟在做Pixiv日榜的收集,回头看看效果怎么样,实在不行就去研究一下各种各样的什么booru之类的图站吧,用那些图片也是个不错的选择。